Latch'ler(mandallar) dijital devrelerde kullanılan bir bellek elemanlarıdır. Adını yaptığı işe benzerliğinden ötürü almıştır. Mandal, yani tutucu demektir. Genellikle, bir sinyalin belirli bir anda sahip olduğu değeri hatırlamak için kullanılır. Latch'ler, özellikle geçici veriyi saklamak amacıyla kullanılır ve flip-flop'lara göre daha basit yapılara sahiptir. Latchler ile flip-floplar arasındaki fark saat (clk) sinyaline bağlı çalışmamalarıdır. Saat sinyalinden bağımsız olarak veriyi tutarlar.

#### Özetle:

- Latch, giriş sinyali aktifken mevcut veri değerini saklar.
- Giriş sinyali değiştiğinde, latch yeni sinyal değerini alır ve saklar.
- Saat sinyaline bağlı olmadığı için sürekli olarak girişe duyarlıdır ve bu yüzden leveltriggered olarak tanımlanır (örneğin, giriş yüksek seviyedeyken saklar, düşük seviyeye geçtiğinde tutmayı bırakır).

Latch'ler, flip-flop'lara kıyasla daha az kararlıdır ve zamanlamaya bağlı hatalara daha yatkındır. Genellikle flip-flop'lardan önceki geçici depolama işlemlerinde kullanılır.

Yine flip-flop'lar gibi farklı türleri bulunmaktadır. En yaygın olarak bilinenleri SR Latch ve D Latch olanlarıdır. Kısaca bahsetmek gerekirse;

**SR** (**Set-Reset**) **Latch:** En temel latch türüdür. "Set" ve "Reset" girişleri vardır. Set girişine 1 verildiğinde latch çıkışı 1 olur, Reset girişine 1 verildiğinde ise çıkış 0 olur. Ancak, her iki giriş aynı anda 1 olduğunda çelişkili bir durum oluşur, bu yüzden bu giriş kombinasyonundan kaçınılır.

**D** (**Data**) **Latch:** Tek girişe sahiptir ve bu girişe verilen veriyi saklar. Girişin aktif olduğu süre boyunca latch, girişteki değeri çıkışa yansıtır ve saklar. Bu latch, SR latch'in sorunlarını gidermek için tasarlanmıştır ve flip-flop'ların temelini oluşturur.

Latch'ler FPGA'da çok fazla tercih edilmezler çünkü sistemde belli başlı belirsizlikle oluşturabilirler fakat kullanıldıkları birkaç alan vardır. Bunlara kısaca değinecek olursak da şöyle söyleyebiliriz;

# 1. Sinyal Tutma (Hold) İşlevi

Latch'ler, sinyal durumunu tutmak (veriyi saklamak) için kullanılır. Örneğin, bir veri hattında sinyalin sabit kalması gereken durumlarda latch'ler kısa süreli veri saklama işlevi görebilir. Bir saat sinyaline bağımlı olmayan latch'ler, belirli kontrol sinyallerine yanıt olarak veriyi saklar ve gerekli durumlarda veri hattını dengede tutar.

#### 2. Metastabilite ve Zamanlama Kontrolü

FPGA'larda senkronize edilmeyen girişlerin veya kontrol sinyallerinin metastabiliteye neden olmaması için latch'ler kullanılabilir. Bu gibi durumlarda latch'ler, geçici sinyal durumlarını dengeleyerek saat sinyaliyle tam senkronize olmayan sinyallerde stabilizasyon sağlar.

## 3. Özel Durumlar İçin Tasarım Araçları (Hold Buffer olarak)

Bazı FPGA tasarımlarında, küçük veri yolu durumu değişikliklerinde gecikme tamponu olarak kullanılabilir. Bu, özellikle belirli gecikme gerektiren veri yollarında veya sinyal yönlendirme sürecinde kullanılan latch'lerin küçük gecikmeler sağlayarak sinyalleri tamponlamasına yardımcı olabilir.

#### 4. Statik Veri Saklama

Latch'ler, belirli bir kontrol sinyaline dayalı olarak veri saklamada statik (durum koruyucu) bir yapı sağlar. Örneğin, FPGA içindeki özel durum makinelerinde (state machines) bir kontrol sinyali değiştiğinde belirli sinyallerin güncellenmeden önce geçici olarak tutulmasını sağlamak için latch'ler kullanılabilir.

Fakt latchler çok sık tercih edilmezler hatta istenmezler bunun da sebepleri vardır bu sebeplerden ötürü de latchlerden kaçılınır. Bunu da şu şekilde anlatabiliriz;

FPGA tasarımlarında genellikle **senkron devreler** tercih edilir. Latch'ler saat darbesi yerine kontrol sinyalleriyle çalıştığı için zamanlama sorunları ve yarış durumlarına (race conditions) yol açabilir. Bu durumlar, devrenin doğru çalışmamasına neden olabileceği için latch kullanımı azaltılır ve genellikle flip-flop gibi daha güvenilir yapı taşları tercih edilir.

Race condition (yarış durumu) kavramını biraz açmak gerekirse de dijital sistemlerde iki veya daha fazla sinyalin veya işlemin aynı anda gerçekleşmeye çalışması ancak farklı

hızlarda sonuçlanmasıyla ortaya çıkan bir belirsizlik veya beklenmedik durumdur. Özellikle paralel çalışan süreçlerde, sinyaller arasındaki zamanlama farklılıkları yarış durumlarına yol açabilir. Bu, sonuçların öngörülemez hale gelmesine neden olur ve devrenin beklenmeyen şekillerde davranmasına yol açabilir.

Vivado gibi sentez araçları genellikle latch oluşturulmasından kaçınır ve tasarımcı istemeden latch oluşmasını engellemeye çalışır. Ancak bazı durumlarda, kod yapısı yanlış yazıldığında veya sinyaller belirli koşullar altında atanmadan bırakıldığında, sentez aracı latch oluşturmaya zorlanabilir. Mesela bir sinyalin tüm durumlarda atanmadığı veya başlangıç değerinin eksik bırakıldığı koşullarda ortaya çıkabilir. Vivado bu durumda değerin tutulması gerektiğini düşünür ve latch yapısı oluşturarak sinyal durumunu korur. Örneğin bir **if-else** veya case yapısında her durumu oluşturup kapsayamazsanız Vivado otomatikmen eksik durumlar için sinyalin tutulması gerektiğine karar verir ve bir latch oluşturur. Bunun önüne geçmek için de her durumu açıkça belirtmek veya varsayılan bir durum eklemek latch oluşumunu engeller.

Ayrıca Vivado gibi sentez araçları kod yapısında saat sinyali içermeyen process'leri(işlemleri) gördüğünde durumları korumak için latch kullanmaya karar verebilir. Özellikle asenkron kontrol sinyalleriyle çalıştığınızda ve sinyal değerlerini sadece bu kontrol sinyaline atadığınız zaman Vivado sinyalin saklanması gerektiğini düşünüp latch oluşturabilir.

Bu yapılardan kaçmak için yapılması gerekenleri özetlememiz gerekirse;

- 1. Sinyal atamaları eksiksiz yapılmalı, her koşulda sinyale bir değer atanmalı,
- 2. Varsayılan durum kullanılmalı, **case** ifadelerinde **when others** => kullanarak her durumun ele alındığından emin olunmalı,
- 3. Saat sinyali kullanılmalı, mümkünse tüm süreçleri saat sinyali ile senkronize edilmeli ve yalnızca gerekli durumlarda asenkron yapılar tercih edilmeli.

Burada sizlere D Latch ile alakalı daha detaylı bilgi verip bununla ilgili bir VHDL kodu yazacağım.

# **D LATCH**

D latch, dijital devrelerde kullanılan önemli bir bellek elemanıdır. "D" harfi "Data"yı temsil eder ve bu latch'in temel amacı, girişteki bir veri bitini geçici olarak saklamaktır. D latch'in çalışma prensibi basittir, tek bir girişi vardır. Çıkışta Q ve Q' (ters Q) bulunur. Q, verinin kendisini, Q' ise verinin tersini temsil eder. Aşağıda giriş çıkışları devre üzerinden görebilirsiniz.

### **Devre Gösterimi**



Çalışma prensibi şu şekildedir:

D latch, giriş sinyaline bağlı olarak level-triggered çalışır, yani Enable (E) girişinin durumuna bağlı olarak veriyi alır veya saklar:

Enable ( $\mathbf{E}$ ) = 1 (aktif): D latch,  $\mathbf{D}$  girişindeki veriyi doğrudan  $\mathbf{Q}$  çıkışına yansıtır. Yani D'deki değişimler Enable sinyali aktifken anında  $\mathbf{Q}$  çıkışında görülür.

Enable ( $\mathbf{E}$ ) = 0 (pasif): D latch,  $\mathbf{D}$  girişindeki veriyi yok sayar ve o anda  $\mathbf{Q}$  çıkışında olan değeri saklar.

Bu sayede D latch, Enable sinyali aktif olduğu sürece girişteki veriyi güncelleyebilir, pasif olduğunda ise çıkışını koruyarak bellek işlevi görür.

Çalışma prensibinin tablo halini de aşağıda görebilirsiniz.

# Doğruluk Tablosu

| E | D      | Q | Tanım                  |
|---|--------|---|------------------------|
| 0 | X(0,1) | Q | Değişiklik<br>yok      |
| 1 | 0      | 0 | Q = 0; reset<br>durumu |
| 1 | 1      | 1 | Q = 1; set<br>durumu   |

Kullanılan bu D Latch'in avantajı olarak da belirsiz durumları önleyeceğini söyleyebiliriz. SR Latch'teki çelişkili durumu ortadan kaldırır. Böylece daha kararlı bir çalışma sağlar. Ayrıca tek bir girişi olduğu için de tek bir veri girişine ihtiyaç duyar. Bu da dijital devrelerde kullanımı basitleştirir.

Şimdi bu D mandalının basitçe VHDL kodunu ve açıklamalarını bir sonraki sayfada bulabilirsiniz.

NOT: VHDL kodunda saat frekansı (clk) kullanılmasının sebebi simülasyon ekranında istenilen durumları rahatça görebilmektir. Yoksa yukarıda bahsedildiği gibi latch saat sinyali kullanılmadığı durumlarda ortaya çıkar. Aşağıdaki tasarım biraz daha flip-flop yapısına uygun bir tasarımdır fakat bizim yaptığımız sadece uygulamayı daha kolay gözlemlemek olduğu içindir. Kafa karışıklığını gidermek adına bu durumu dikkate alınız.

```
library IEEE;
use IEEE.STD LOGIC 1164.ALL;
use ieee.numeric std.all;
use ieee.std logic unsigned.all;
entity dlatch is
    Port ( D_i : in STD_LOGIC;

E_i : in STD_LOGIC;

clk_i : in STD_LOGIC;

rst_i : in STD_LOGIC;

Q_o : out STD_LOGIC
            );
end dlatch;
architecture Behavioral of dlatch is
begin
    process (clk i, D i, E i)
         variable DE i : std logic vector (1 downto 0);
         variable latch : std logic:='0';
         variable cikis : std logic;
         begin
              DE i:= D i & E i;
                  if (rst i = '1') then
                       Q o <= '0';
                  elsif rising_edge (clk_i) then
                       case DE_i is
                          when "00" => cikis := latch;
                          when "01" => cikis := latch;
                          when "10" => cikis := '0';
                          when "11" => cikis := '1';
                          when others => null;
                       end case;
                 end if;
         Q_o <= cikis;
    end process;
end Behavioral;
```

Kodu detaylı olarak incelersek;

**dlatch** isimli bu entity, dış dünyaya açılan giriş ve çıkışları tanımlar. Giriş ve çıkış sinyalleri şunlardır:

**D\_i:** Veri girişi sinyali. Bu, latch'in saklayacağı ya da Q o çıkışına göndereceği

veriyi sağlar.

**E\_i:** Enable ya da etkinleştirme sinyali. E sinyali '1' olduğunda, latch D girişini saklayabilir veya D değerine göre çıkışı değiştirebilir.

**clk\_i:** Saat sinyali. Latch çıkışı bu saat sinyaliyle tetiklenir; rising\_edge (pozitif kenar) olduğunda çıkış güncellenir.

rst\_i: Reset sinyali. Reset aktifken, devre çıkışı (Q o) sıfırlanır.

**Q\_o:** Çıkış. Latch'in tutmakta olduğu değeri dış dünyaya iletir.

```
begin
    process (clk_i, D_i, E_i)
    variable DE_i : std_logic_vector (1 downto 0);
    variable latch : std_logic:='0';
    variable cikis : std_logic;
```

Mimari
tanımında,
process bloğu
clk\_i, D\_i ve E\_i
sinyallerini

kullanır. Bu işlem bloğunda üç değişken tanımlanmıştır:

**DE\_i**: **D\_i** ve **E\_i** sinyallerini birleştirerek 2 bitlik bir vektör oluşturur.

latch: Latch'in önceki durumunu saklayan değişken, başlangıç değeri '0' olarak tanımlanmıştır.

cikis: Q\_o çıkışına atanacak geçici değer.

Bu değişkenlerle, devrenin rst\_i ve DE\_i durumlarına göre çıkışı nasıl etkileyeceği belirlenir.

```
begin
    DE_i:= D_i & E_i;
```

Bu satır, **D\_i** ve **E\_i** sinyallerini birleştirerek **DE\_i** adlı 2 bitlik bir vektör elde eder.

rst\_i (reset) sinyali
'1' olduğunda, Q\_o
çıkışına sıfır atanır. Bu,
devrenin başlangıç
durumuna geri
döndüğünü gösterir.

Reset aktif olmadığında ve **clk\_i** sinyali pozitif kenara geldiğinde, **DE\_i** durumuna göre cikis değişkeni belirlenir:

- "00": **D\_i** ve **E\_i** ikisi de '0' olduğunda, çıkış latch'te tutulan önceki değeri (eski değeri) korur.
- "01": **D\_i** '0', **E\_i** '1'

olduğunda da yine çıkış eski değeri korur.

- "10": **D\_i** '1', **E\_i** '0' olduğunda çıkış sıfırlanır.
- "11": **D\_i** ve **E\_i** ikisi de '1' olduğunda çıkış bir yapılır.

Bu, **DE\_i** durumlarına göre çıkışın nasıl etkileneceğini belirleyen temel çalışma prensibini ortaya koyar.

En son satırda, **cikis** değişkeninin değeri **Q\_o** çıkışına atanır. Bu, saat sinyalinin pozitif kenarında güncellenmiş **cikis** değerini **Q\_o** çıkışına aktarır.

Bir sonraki sayfada ise yazdığımız kodun testini gerçekleştirmek üzere Testbenc kodunu inceleyelim.

```
library IEEE;
use IEEE.STD_LOGIC_1164.ALL;
entity tb dlatch is
end tb dlatch;
architecture Behavioral of tb dlatch is
component dlatch is
     Port (
            D_i
                     : in STD_LOGIC;
            E
                   : in STD_LOGIC;
: in STD_LOGIC;
: in STD_LOGIC;
: out STD_LOGIC
                      : in STD LOGIC;
            clk_i
            rst_i
            Q o
            );
end component;
                     : std_logic := '0' ;
signal SD i
                  : std_logic := '0';
: std_logic := '0';
: std_logic := '0';
: std_logic := '0';
signal SE i
signal Sclk_i
signal Srst_i
signal SQ o
constant clk period :time := 10ns;
begin
uut: dlatch port map (
                           Di
                                  => SD_i,
                           Εi
                                  => SE i,
                           clk_i => Sclk_i,
                           rst_i => Srst_i,
                           Q_0
                                  => SQ o
                       );
clk i process: process
                           Sclk_i <= '0';
                           wait for clk_period/2;
                           Sclk i <= '1';
                           wait for clk_period/2;
                  end process;
--Test
ED in: process
             begin
                  Srst_i <= '1';
                  wait for clk_period*2;
Srst_i <= '0';</pre>
                  wait for clk_period*2;
                  SE_i <= '0';
                  SD_i <= '0';
wait for clk_period*2;</pre>
                  SE_i <= '0';
                  SD i <= '1';
                  wait for clk_period*2;
                  SE_i <= '1';
                  SD_i <= '0';
                  wait for clk_period*2;
                  SE i <= '1';
                  SD_i <= '1';
                  wait for clk period*2;
         end process ED in;
end Behavioral;
```

Detaylı incelemesini aşağıda yapalım:

```
library IEEE;
use IEEE.STD_LOGIC_1164.ALL;
entity tb_dlatch is
end tb_dlatch;
```

Bu kısımda, tb\_dlatch adlı testbench entity tanımlanır. Testbenchlerde genellikle giriş veya çıkış portları olmaz, çünkü testbench doğrudan sinyal ve işlemler tanımlar ve modülün çalışmasını test etmek için kendi içerisinde sinyalleri yönlendirir.

Bu kısımda, test edilecek olan dlatch bileşeni tanımlanır. **dlatch** bileşeninin giriş ve çıkış portları, daha önceki dlatch kodunda tanımlananlarla aynıdır. Aslında ara bağlantıları burada yaparız:

- **D\_i:** Latch'e girilecek veri.
- **E\_i**: Latch'in etkinleştirme sinyali.
- clk\_i: Saat sinyali.
- rst i: Reset sinyali.
- **Q\_o:** Latch'in çıkışı.

Bu kısımda dlatch bileşeninin portlarına bağlı olan sinyaller tanımlanır. Bu sinyaller, test sırasında bileşene uygulanacak olan giriş ve çıkışları temsil eder:

SD\_i, SE\_i, Sclk\_i, Srst\_i: D\_i,
 E\_i, clk\_i, ve rst\_i portlarına bağlanan sinyallerdir.

• **SQ\_o**: **Q\_o** çıkışına bağlanır.

**clk period** ise saat sinyali için kullanılan bir zaman sabitidir ve 10ns olarak tanımlanmıştır.

Bu satırda, **uut** (unit under test) adı verilen dlatch bileşeni testbench'e dahil edilir ve giriş/çıkışları ilgili sinyallere bağlanır. **uut**, **SD\_i**, **SE\_i**, **Sclk\_i**, **Srst\_i** ve **SQ\_o** sinyallerine bağlanarak test edilecek olan dlatch bileşeninin tb\_dlatch üzerindeki portlara erişmesini sağlar.

Bu işlem, saat sinyalini üretir ve clk\_period süresine göre her yarı periyotta bir sinyalin durumunu değiştirir (0 ve 1). Yani, burada 10ns periyot tanımlandığı için her 5ns'de bir durum değiştiren bir saat sinyali üretilmiş olur. Bu saat sinyali, dlatch bileşeninin clk\_i girişini sürmek için kullanılır.

```
ED in: process
                Srst i <= '1';
                wait for clk period*2;
                Srst i <= '0';
                wait for clk_period*2;
                SE_i <= '0';
                SD i <= '0';
                wait for clk period*2;
                SE_i <= '0';
                SD i <= '1';
                wait for clk_period*2;
                SE i <= '1';
                SD i <= '0';
                wait for clk period*2;
                SE i <= '1';
                SD i <= '1';
                wait for clk_period*2;
        end process ED in;
end Behavioral;
```

Bu işlem bloğu, dlatch devresinin giriş sinyallerini belirli bir sırayla değiştirerek testler uygular. Her bir durumda, clk\_period\*2 kadar beklenir ve ardından yeni bir giriş kombinasyonu uygulanır:

- 1. **Başlangıçta reset aktif edilir: Srst\_i** '1' yapılır ve iki saat periyodu boyunca beklenir. Bu sırada **Q\_o** çıkışı sıfırlanır.
- 2. **Reset devre dışı bırakılır: Srst\_i** '0' yapılır ve yine iki saat periyodu beklenir. Bu adımda, latch devresi artık girişlere göre davranmaya başlar.

- 3. İlk test kombinasyonu: SE\_i = 0, SD\_i = 0. Latch devresi etkin değilken D sinyali sıfırdır.
- 4. **İkinci test kombinasyonu: SE\_i** = 0, **SD\_i** = 1. Latch devresi etkin değilken D sinyali bir olur.
- 5. Üçüncü test kombinasyonu: SE\_i = 1, SD\_i = 0. Latch devresi etkinken D sinyali sıfır olur.
- 6. **Dördüncü test kombinasyonu: SE\_i** = 1, **SD\_i** = 1. Latch devresi etkinken D sinyali bir olur.

Bu giriş kombinasyonları, latch'in  $\mathbf{D}$  ve  $\mathbf{E}$  girişlerine göre  $\mathbf{Q}_{\mathbf{0}}$  çıkışında doğru davranışı sergileyip sergilemediğini test etmek için kullanılır.

Yazdığımız bu test sonrasında aşağıda gözüken simülasyon ekranımız karşımıza çıkmaktadır. Buradaki ekrandan sinyalleri kontrol ederek doğru yaptığımızı da kontrol edebiliriz.



