

# STM32 ile Gömülü Yazılım

5 Mayıs 2021 Çarşamba 07:50

[01 GPIO](#)

[02 EXTI](#)

[03 ADC](#)

[04 DAC](#)

[05 DMA](#)

[06 TIMER](#)

[07 PWM](#)

[08 USART](#)

[09 I2C](#)

[10 SPI](#)

[11 CAN](#)

## [Uygulamalar](#)

[01\\_01 Harici Led Yakma](#) [HAL, REGISTER]

[01\\_02 Buton ile Led Yakma](#) [HAL, REGISTER]

[02\\_01 External Interrupt](#) [HAL, REGISTER]

[03\\_01 ADC Verisi Okuma](#) [HAL, REGISTER]

[03\\_02 ADC Interrupt](#) [HAL]

[04\\_01 DAC Kullanımı](#) [HAL, REGISTER]

[04\\_02 ADC Değeri İle DAC Kontrolü](#) [HAL]

[05\\_01 DMA ile ADC Değer Okuma](#) [HAL, REGISTER]

[06\\_01 Timer Değer Okuma](#) [REGISTER]

[06\\_02 Timer Interrupt](#) [HAL]

[07\\_01 PWM Kullanımı](#) [HAL, REGISTER]

[08\\_01 USART ile Mesaj Gönderme](#) [REGISTER]

[09\\_01 I2C Kullanımı](#) [REGISTER]

[10\\_01 SPI Kullanımı](#) [HAL]

[11\\_01 CAN Kullanımı](#) [HAL]

# Giriş

5 Mayıs 2021 Çarşamba 08:02

## Giriş

### Kaynaklar

- Bu belge oluşturulurken <https://www.udemy.com/course/stm32f4-discovery-kart-ile-arm-dersleri/> linkteki eğitim kursu izlenirken alınan notlardan oluşmaktadır.

# 01 GPIO

5 Mayıs 2021 Çarşamba 08:02

## 01 GPIO

### Giriş

- Butonlar ve anahtarlar mikrodenetleyiciye giriş pini üzerinden lojik 1 ve lojik 0 olarak bilgi girişini sağlayan mekanik elemanlardır.



USER & WAKE-UP Button

- Resimde görüldüğü gibi kullanıcı butonu A portunun 0. pinine bağlı ve pull down durumundadır.



- PullUp bağlantıda GPIO girişi direnç üzerinden + beslemeye (VCC/VDD) bağlanır. Butona basılmadığı durumda GPIO girişinde lojik 1 vardır. Butona basıldığı durumda girişe OV (lojik 0) uygulanmış olur.
- PullDown bağlantıda, GPIO girişi direnç üzerinden GND ye bağlanır. Butona basılmadığı durumda girişte lojik 0 bulunur. Butona basıldığı durumda buton üzerinden lojik 1 uygulanmış olur.





- Buton ve anahtarda konum değiştiğinde arktan dolayı mikrodenetleyici girişinde çok sayıda istenmeyen lojik değer oluşur. Bu duruma ark deniyor.



## 02 EXTI

5 Mayıs 2021 Çarşamba 08:02

### 02 EXTI

#### Giriş

- Önceliği yüksek işlerin mikrodenetleyici tarafından ana program akışını keserek yapılmasına interrupt (kesme) denir.
- Eğer bir kesme kaynağından mikrodenetleyiciye uyarı gelirse mikrodenetleyici yapmakta olduğu işi bekletir, kesme alt programına gider, o programı icra eder, daha sonra ana programda kaldığı yerden devam eder.
- Kesmeleri genellikle çok hızlı yapılması gereken işlemlerde, anlık tepki verilmesi gereken yerlerde kullanırız.
- Harici bir kaynaktan oluşan olaylardan dolayı meydana gelen kesmelere, harici kesmeler denir.  
Harici kaynak olarak, dış ortamdan pinler vasıtıyla gelecek kesme ve kandi içindeki donanımlardan gelen kesmeleri anlayabiliriz.
- STM32F407 mikrodenetleyicisi için porttaki 0.pin EXTI0 kanalına bağlıdır.



- Bunlar dışında 7 tane daha kanal vardır. Toplamda 23 kanal vardır.

EXTI line 16 is connected to the PVD output

EXTI line 17 is connected to the RTC Alarm event

EXTI line 18 is connected to the USB OTG FS Wakeup event

EXTI line 19 is connected to the Ethernet Wakeup event

EXTI line 20 is connected to the USB OTG HS (configured in FS) Wakeup event

EXTI line 21 is connected to the RTC Tamper and TimeStamp events

EXTI line 22 is connected to the RTC Wakeup event

- Karmaşık kesme isteklerinin işlemciye sürekli yük getirmemesi için işlemci içerisinde özel bir donanım bloğu oluşturulmuştur. Bu donanıma kesme kontrolörü (interrupt controller) adı verilir.
- Kesme kontrolörü haklı bir sebeple gelen kesme isteği neticesinde düzgün işleyen programı askiya alarak kesme fonksiyonu (interrupt function) olarak adlandırılan özel kod parçasını işlemeye başlar.
- Kesme fonksiyonunun işletilmesinin bitiminde program kaldığı yerden çalışmaya devam eder.
- NVIC kontrolör mikroişlemci içerisindeki önemli donanım kesmelerini (DMA istekleri, USART, CAN, I2C ve Timer gibi donanım kesmeleri) ve ayrıca External Interrupt/Event Controller (EXTI) adı verilen donanım vasıtasyyla portlardan gelen kesmeleri kontrol eder.



# 03 ADC

5 Mayıs 2021 Çarşamba 08:02

## 03 ADC

### Giriş

- Doğada var olan bütün fiziksel büyüklükler (ısı, ışık, ses, zaman vs.) analog büyüklik kavramına girer.
- Dünyadaki herhangi bir şeyi dijital sistemlerimiz ile ölçmek, değerlendirmek, işlemek ve bu değerlere göre işlem yapabilmek için ADC (Analog Digital Converter) ihtiyaç vardır.

- ADC modülleri gerek harici, gerek dahili olsun hepsi bir referans voltaja ihtiyaç duyarlar.
- Genellikle mikroişlemcilerde referans voltajı işlemcinin besleme gerilimidir.
- Bu değer aynı zamanda ayarlar yapılarak harici olarak verilebilir.
- ADC'ler 10, 12, 16, 24 vb. bit çözünürlükte bulunurlar.
- STM32F4 discovery kartı üzerinde bulunan ADC ler 6,8,10 ve 12 bit çözünürlükte çalışabilirler.
- STM32F4 discovery kartında referans voltajı default 3.3V dur.
- ADC modülünün 10 bit olduğunu düşünelim.  $2^{10} = 1024$  değeri okunacak maksimum değerdir.
- Yani ADC okuması yaparken  $0V = 0$ ,  $3.3V = 1023$  değeri ile bize döner.
- Buradan her bit bir değerinin alacağı voltaj değerini ( çözünürlüğü )  $3.3 / 1024 = 0,00322265625$  olarak buluruz.
- Buradan da biz ADC modülünden okuduğumuz değeri bu ifade ile çarparak voltaj değerini buluruz. Örneğin 640 okuduysak  $640 * 0,00322265625 = 2,0625$  V olduğunu buluruz.
- STM32F407VG işlemcisinde 3 adet ADC birimi mevcuttur.
- ADC biriminin ulaşabileceği maximum hız 36 MHz dir. Bu hız aynı zamanda ADC çözünürlüğü ile ters orantılıdır.
- ADC biriminin ölçüleceği en düşük gerilim değişimi (hassasiyet) çözünürlük denir.
- Çözünürlük arttıkça ADC biriminin ölçüm hızı düşmektedir.
- STM32F407VG mikrodenetleyicisinin en yüksek ADC çözünürlüğü 12 bittir.

### CÖZÜNÜRLÜK

### ADC ÇEVİRİM HIZI

|        |          |
|--------|----------|
| 12 Bit | 12 Cycle |
| 10 Bit | 10 Cycle |
| 8 Bit  | 8 Cycle  |
| 6 Bit  | 6 Cycle  |

- STM32F4 mikrodenetleyicisinde 0V-3.6V aralığında ölçümler yapılmaktadır.
- Buradaki voltaj aralığında ADC biriminin beslemesi ( $V_{DDA}-V_{SSA}$ ) ile ilgili bir durumdur.
- ADC biriminin besleme voltajı ve referans gerilimi ( $V_{REF+}$ ,  $V_{REF-}$ ) ADC biriminin ölçüleceği gerilim aralığını belirler.
- Her ne olursa olsun ADC birimi 3.6V dan fazlasını ölçemez.

- Analog bir değerden dijital bir değere dönüşüm yapılırken dikkat edilmesi gereken hususlar vardır.
- Bunlardan en önemlisi, ölçülecek analog gerilim ~~değerinin~~ dönüşümü yapacak çipin ölçüm aralığında olması gerekir.
- Diğer önemli nokta, ölçüm yapılacak hassasiyetin belirlenmesi ve buna uygun bit genişliğinde bir dönüştürücü seçilmelidir.
- Ölçüm hassasiyetinde önemli olan dönüşüm yapacak sistemin bit çözünürlüğüdür(Resolution).
  - $\text{Resolution} = V_{ref+}/(2^n - 1)$
- Örneğin 8 bit çözünürlüğe sahip ve 0 - 3.3V aralığında arası ölçüm yapabilen bir ADC ölçüm ünitesinin ölçüleceği minimum değer (hassasiyet) yaklaşık olarak 12mV dur.
  - $\text{Resolution} = 3.3 / (2^8 - 1) = 3.3 / 255 = 0,01294... \approx 12\text{mV}$
- 12 bit çözünürlüğe sahip ve 0 - 3.3V aralığında arası ölçüm yapabilen bir ADC ölçüm ünitesinin ölçüleceği minimum değer (hassasiyet) yaklaşık olarak 805uV dur.
  - $\text{Resolution} = 3.3 / (2^{12} - 1) = 3.3 / 4095 \approx 805\text{uV}$

**1- Tek Dönüşüm Modu:** Sadece bir kere analog sinyali dijital sinyale çevirir.

**2- Sürekli Dönüşüm Modu:** Her dönüşüm işlemi bittiğinde, yeni bir dönüşüm işlemeye başlar. Yani sürekli olarak ~~analog~~ sinyalımızı dijital olarak çıktısını verir.

**3- Tarama Modu:** Bu mod ise birden fazla kanalda ADC çevrimi yaptığımız zaman, seçtiğimiz kanallarda tarama yaparak bu kanallarda dönüşüm yapmaya çalışır. Eğer bu modu sürekli mod gibi çalıştırırsak, seçtiğimiz kanalları sürekli olarak tarar. Aksi takdirde seçtiğimiz kanallar bir kere taranır ve durur.

**4- Süreksiz Mod:** Bu modda ise kanallar üzerinden çevrim sayınız 8 den az olmalıdır.

# 04 DAC

5 Mayıs 2021 Çarşamba 08:02

## 04 DAC

### Giriş

- DAC, dijital sinyali analog sinyale çevirme işlemi yapan birimdir.
- STM32F4 discovery kartında 0 - 3.3 V arasında tüm gerilimleri çıkış olarak vermemizi sağlar.
- STM32F407VG mikrodenetleyicisi içerisinde dahili olarak 12 bit tampona sahip, iki adet DAC birimi bulunur. Bu birimler sayesinde dijital bir veriyi analog bir veriye dönüştürerek çıkış üretilebilir.
- STM32F407VG'ye ait DAC birimleri 8 bit veya 12 bit değerinde çıkış üretilebilirler.
- 12 bit değerinde kullanılırken, veri 16 bitlik kaydedici içerisinde sola veya sağa dayalı şekilde kullanılabilir.
- DAC biriminin önemli özelliklerinden bir tanesi, gürültü veya sinyali üretebilme özelliğidir.
- Üçgen dalga üretebilme özelliğine sahiptir.
- Dönüşüm işlemi için harici tetikleyiciler kullanılabilir.
- DMA ile kullanılabilir.
- PA4 ve PA5 pinleri üzerinden çıkış verilmektedir.
- DAC birimleri APB1 veri yoluna bağlıdır, kullanmak için aktif etmek gereklidir.
- 12 bitlik çözünürlüğe sahip bir DAC biriminin referans gerilimleri  $V_{SSA} = 0 \text{ V}$ ,  $V_{DDA} = +3 \text{ V}$  ele alınır ise, adım başına üreteceği voltaj şu şekilde hesaplanır:
  - >  $\text{DACout} = \text{Vref} / 4095$
- Buradan adım başına düşen voltaj:
  - >  $\text{DACout} = 3\text{V}/4095 = 732,600732\dots$
- Buradan istenilen voltajı elde etmek için istenilen voltaj değeri adım başına üretilen voltaj değerine bölünerek elde edilir.
  - > Örneğin 1V elde etmek isteniyorsa:
    - $1/0,000732600\dots = 1365$  değeri elde edilir.

# 05 DMA

5 Mayıs 2021 Çarşamba 08:03

## 05 DMA

### Giriş

- Çeşitli çevre birimlerinden okuduğumuz verileri bir değişkene atarız. Bu değişkenler RAM'de depolanır. Bu işlem normalde çevre birimlerinde okunan verinin CPU'ya alınıp ardından RAM'e yazılır. Ancak CPU kullanımını hem işlemciyi yorar hemde kayıplara yol açar. DMA (Doğrudan Bellek Erişimi) sayesinde verileri direk olarak RAM'e yazma imkanı buluruz.
- DMA donanımı CPU'dan bağımsız olarak verilerimizi peripheral (çevresel birim) den hafızaya, hafızadan çevresel birime ve hafızadan hafızaya olmak üzere hızlı bir şekilde kaynak adresinden hedef adrese aktarır.
  - > peripheral -> memory
  - > memory -> peripheral
  - > memory -> memory
- Bu sayede CPU'nun yükünü hafifletmiş oluruz. Sistem sanki 2 CPU ile çalışıyordu gibi düşünülebiliriz.
- Örneğin bilgisayarlarımıza bulunan 4 gerçek 4 sanal çekirdekteki sanal, aslında DMA diyebiliriz.
- DMA isteği için çevresel birim tarafından (ADC, DAC, I2c vs) DMA kontrolcüsüne istek gönderilir, kontrolcüde bu isteğin sırası gelince ilgili çevresel birime geri bildirimde bulunur ve işlem kaynak adresinden hedef adrese doğru gerçekleşir.
- STM32F4 ' te iki adet DMA vardır.
- DMA1'in DMA 2'den kanal 1'in kanal 2'den yüksek olduğu bilinmektedir.
  - > Öncelik sırası belirtmek için dört seviye vardır. Low, Medium, High, Very High.
- Aynı anda birçok kanal kullanıldığından hangi kanalın öncelik değeri fazla ise ilk o kanal alınır.
- DMA'lar paralel olarak çalışmazlar, seri olarak çalışırlar. Bu nedenle hangisinin sırası geldi ise o anda o çalışır.
- İlk olarak ADC den okuduğumuz değerleri, hiç CPU ya uğratmadan DMA ile direkt RAM'e yazacağız.
- Discovery boardda 2 adet DMA vardır ve ADC'yi DMA1 desteklemez, DMA2 destekler. Bu nedenle çalışma işlemi yaparken DMA2'yi kullanacağız.

# 06 TIMER

5 Mayıs 2021 Çarşamba 08:02

## 06 TIMER

### Giriş

- Timer modülünün temel görevi zamanlama yapmaktadır.
- İşlemci frekanasına bağlı olarak çalışırlar.
- Dışarıdan gelen pulse (darbe)leri sayarlar.
- İşlemciye tanıtılan bir süre ile, geçen süreyi karşılaştırma ve belli bir süre sonunda kesme üretme gibi işlemlerde kullanılırlar.
- Çoğu mikrodenetleyicide PWM(Pulse Width Modulation) birimleri de Timer ünitelerine bağlı olarak çalışırlar.
- STM32F4 DISCOVERY işlemcisinde toplam 17 adet timer bulunur.
- 10 adet genel amaçlı timer
- 2 adet gelişmiş timer
- 1 adet IWDG (Independent Watchdog) timer
- 1 adet WWDG (Window Watchdog) timer
- 1 adet Systemtick (Sistem Zamanlayıcısı) timer var.

| Timer                                                  | Özellik                   |
|--------------------------------------------------------|---------------------------|
| TIM1ve,TIM8                                            | 16 bit gelişmiş timer     |
| TIM3, TIM4, TIM9, TIM10,<br>TIM11, TIM12, TIM13, TIM14 | 16 bit genel amaçlı timer |
| TIM2, TIM5                                             | 32 bit genel amaçlı timer |
| TIM6, TIM7                                             | 16 bit basit timer        |

## **1- Genel Amaçlı Timer2, Timer3, Timer4 Ve Timer5 Birimleri**

- Timer 2,3,4 ve timer5 birimleri, düşük hızlı APB1 (42 MHz) veri yolu üzerinde bulunmaktadır. Eğer APB1 prescaler (bölgücü) değeri 1 den farklı ise bu timerların clock frekansları beslendikleri frekansların 2 katına çıkar. Yani 84 MHz clock frekansına sahip olur.
- Timer3(TIM3) ve Timer4(TIM4) 16-bit lik sayıcıya, Timer2(TIM2) ve Timer5(TIM5) 32-bit lik sayıcıya sahiptirler.
- Bu sayıcılar (up)yukarı, (down)aşağı ve (auto-reload)hizalanmış/merkezlenmiş modlarda sayma yapabilirler.
- Ayrıca bu sayıcıların otomatik yükleme özellikleri de vardır.
- 16-bit genişliğinde kontrol edilebilir ön bölgücü değeri (prescaler) vardır.
- Bu timer biriminde 4x16 adet yüksek çözünürlüktü capture/compare kanalı bulunur. Bu kanallar;
  - > Giriş Yakalama (Input Capture)
  - > Çıkış Karşılaştırma (Output Compare)
  - > Darbe Genişlik Modülasyonu (PWM)
  - > Tek Darbe Çıkışı (One-Pulse)
- Dahili diğer Timer birimleri ile sonkronizasyon
- Interrupt (kesme) ve DMA üretimi
- Clock kaynağı seçimi

## **2- Basic(Basit) Timer Timer6 ve Timer7**

- Timer 6 ve Timer 7 birimleri birbirinden tamamen bağımsızdır. Ortak register veya ortak veri kullanımı gibi bir durum yoktur.
- Basic Timer birimleri genel sayaç olarak kullanılabilecekleri gibi, spesifil olarak Dijital Analog Çevirici (DAC) biriminin tetikleyicisi olarak da kullanılabilir.
- 16-bit genişliğinde otomatik geri yüklenen artan sayaca sahiptir. (auto-reload upcounter)
- 16-bit genişliğinde kontrol edilebilir ön bölgücü(Prescaler) değere sahiptir.
- DAC birimi için tetikleme çıkışlarına sahiptir.
- Interrupt ve DMA üretimi mevcuttur.
- Çalışma prensibi genel amaçlı timer'ların çalışma prensibi ile aynıdır.

## **3- Gelişmiş Timer 1 Ve Timer 8 Birimleri**

- Timer 1 ve Timer 8, yüksek hızlı APB2 veri yolu (84 MHz) üzerinde bulunurlar. Eğer APB2 prescaler değişkeni "1" değerinden farklı ise bu timer birimlerinin saat frekansı, APB2'nin frekans değerinin iki katı olur. Yani, bu timer birimlerinin maksimum çalışma frekansları 168 MHz olabilir.
- Timer 1 ve Timer 8 birimleri 16 bitlik sayıcıya sahiptirler.
- Bu sayıcılar; yukarı, aşağı ve merkezlenmiş modlarda sayma yapabilirler.
- Bu sayıcıların otomatik geri yükleme özellikleri bulunmaktadır.
- Bu timer birimlerinde 4x16 adet yüksek çözünürlüklü capture/compare kanalı da bulunur.
  - > Bu kanallar giriş öklär olarak ayarlanabilir, çıkış karşılaştırabilir, PWM sinyali üretebilir, sinyal yakalayabilir ve harici bir PWM sinyalini algılayabilirler.

#### **4- Genel Amaçlı 2 Kanallı Timer 9 ve Timer 12 Birimleri**

- Timer 9 yüksek hızlı (84 MHz) APB2 ve Timer 12 düşük hızlı (42 MHz) APB1 üzerinde bulunmaktadır.
- Bu birimlerin frekansları diğerlerinde olduğu gibi veriyolu hızlarının iki katında çalışabilirler.
- Timer 9 ve Timer 12 birimleri 16 bitlik sayıcıya sahiptirler. Bu sayıcılar sadece yukarı sayma yapabilirler. Ayrıca bu sayıcıların otomatik geri yükleme özellikleri de bulunmaktadır.
- Bu timer birimlerinde 2x16 adet yüksek çözünürlüklü capture/compare kanalı da bulunur.
  - > Bu kanallar giriş öķiš olarak ayarlanabilir, çıkış karşılaştırabilir, PWM sinyali üretebilir, sinyal yakalayabilir ve harici bir PWM sinyalini algılayabilirler.

#### **5- Genel Amaçlı 1 Kanallı Timer 10-11 ve Timer 13-14 Birimleri**

- Timer 10 ve Timer 11 yüksek hızlı (84 MHz) APB2 ve Timer 13 ve Timer 14 düşük hızlı (42 MHz) APB1 üzerinde bulunmaktadır. Bu birimlerin frekansları diğerlerinde olduğu gibi veriyolu hızlarının iki katında çalışabilirler.
- Bu birimler 16 bitlik sayıcıya sahiptirler. Bu sayıcılar sadece yukarı sayma yapabilirler. Ayrıca bu sayıcıların otomatik geri yükleme özellikleri de bulunmaktadır.
- Bu timer birimlerinde 2x16 adet yüksek çözünürlüklü capture/compare kanalı da bulunur.
  - > Bu kanallar giriş öķiš olarak ayarlanabilir, çıkış karşılaştırabilir, PWM sinyali üretebilir, sinyal yakalayabilir ve harici bir PWM sinyalini algılayabilirler.

#### **6- INDEPENDENT WATCHDOG(IWDG) BİRİMİ**

- IWDG, işlemci saatinden bağımsız, kendine ait dahili RC osilatörden ( LSI 32 KHz) beslenen bir watchdog timeridir.
- Watchdog kelimesinin Türkçe karşılığı, bekçi köpeği demektir.
- Mikrodenetleyici içerisindeki amacı da bekçilik yapmaktadır.
  - > Peki neye bekçilik yapacak?
- Mikrodenetleyici, harici sebeplerden veya kodlardaki bir hata sebebiyle kilitlenebilir. Mikrodenetleyici kilitlendiğinde, yürütüğü işlemler durur. Bu tür durumlarda mikrodenetleyicinin tekrar başlatılması gereklidir. İşte watchdog timerlar burada devreye girerler. Watchdog timerlarda belirlenen bir süre sonunda sıfırlanırlar ve işlemciyi resetlerler.

#### **7- WINDOW WATCHDOG(WWDG) BİRİMİ**

- WWDG(Pencere Watchdog) birimi belirli bir pencere içerisinde counter (sayıcı) kaydedicisine tekrar değer yüklenebildiği için bu isimle anılmaktadır.
- Ayarlanabilir süre penceresine sahiptir.
- Anormal erken ve anormal geç uygulama davranışını algılayabilir.
- Önceden belirlenen duruma göre işlemciyi resetler.

# 07 PWM

25 Haziran 2021 Cuma 23:48

## 07 PWM

### Giriş

- Pulse Width Modulation (Darbe genişlik modülasyonu); bir kare dalga sinyalinin, yüksek seviyede kalma süresine müdahale ederek, bu sinyalin geriliminin ortalama değerinin değiştirilmesi olarak tanımlanabilir.
- PWM endüstride iletişim, motor kontrol, ısıtma, aydınlatma gibi önemli bir çok alanda kullanılmaktadır.
- Aşağıdaki şikilde sinyalin görev yapan kısmın (duty cycle) süresinin değiştirilmesi, yani PWM olayını göstermektedir.



1. **Mod 1:** Yukarı doğru sayarken CNT < CCRx (Capture Compare Register) dan düşükse kanal aktif, diğer durumda pasif olur. Aşağı doğru sayarken CNT > CCRx ise kanal pasif, değilse aktif olur.
  2. **Mod 2:** Yukarı doğru sayarken CNT < CCRx (Capture Compare Register) dan düşükse kanal pasif, diğer durumda aktif olur. Aşağı doğru sayarken CNT > CCRx is kanal aktif, değilse pasif olur.
- PWM frekansını hesaplamak için, aşağıdaki formüllerden yararlanmamız lazım;
    - >  $\text{Period} = (\text{Timer_Tick_Freq} / \text{PWM_Freq}) - 1$
    - >  $\text{PWM_Freq} = \text{Timer_Tick_Freq} / (\text{Period} + 1)$
    - >  $\text{Timer_Tick_Freq} = \text{Timer_CLK} / (\text{Prescaler} + 1)$
  - Buradan şunu düşünmeliyiz. Timer frekansı kullanıcı tarafından belirlenir. Aynı zamanda PWM de istenilen frekansta çalışılacağı düşünülecek olursa, bizim belirleyeceğimiz iki değer var. Bunlardan biri prescaler, diğeri ise period. Aslında temel olarak PWM in istenilen frekansta çalışması için prescaler değeri küçük bir değer seçilir ve period bu değere göre ayarlanır.

# 08 USART

5 Mayıs 2021 Çarşamba 08:03

## 08 USART

### Giriş

- <https://www.ercankocclar.com/2018/04/uart-iletisim-protokolu-ve-mikroc-kutuphanesi/>

- USART (Universal Synchronous - Asynchronous Receiver - Transmitter) 1 ve 0 lardan oluşan verileri, iki dijital sistem arasında alıp-verme işleminde kullanılan bir iletişim protokolüdür.
- Dijital sistemlerde iletişim paralel ve seri olmak üzere iki türlü yapılır. Paralel iletişimde bilgi vericiden alıcıya aynı anda birden fazla bit gidecek şekilde gönderilir. Böylece tek hamlede birden fazla veri karşı tarafa gönderildiği için iletişim hızı yüksektir. Fakat bu iletişim türünde kullanılan hat sayısı fazladır ve uzun mesafeler için uygun değildir.
- Seri iletişimde ise vericiden alıcıya gönderilecek bilgi, tek hat üzerinden sırayla gönderilir. Bu şekilde giden bilginin, tek hamlede tek biti gönderebileceği için iletişim hızı yavaşlatır. Fakat seri iletişimde hat sayısı azdır ve uzun mesafeli iletişim için daha uygundur.



- USART protokolü bir seri iletişim protokolüdür. Usart protokolünde veriler senkron veya asenkron olarak alınabilirler. Senkron veri alışverişinde bir data hattı ve bir clock hattı bulunmalıdır. Daha hattından gidecek veriler, clock hattından gönderilen sinalın her düşen veya yükselen kenarında alıcıya ilettilir.
- Asenkron modda ise verilerin iletilmesinde bir clock hattına ihtiyaç duyulmaz. Verilerin gönderilmeye başlayacağı, alıcıya bir başlangıç (start) sinyali ile bildirilir ve hemen arkasından veriler akmeye başlar. Burada verilerin gönderim ve alım hızının zamanlanması, alıcının her gelen biti algılayabilmesi açısından çok önemlidir.
- Asenkron seri iletişimde verinin gönderilmeye başlayacağını bildiren bir start biti, verinin gönderilmesinin bittiğini belirten bir stop biti, bir de verilerin doğru olarak gönderilip gönderilmediğini anlamak için kullanılan "parity (eşitlik)" biti bulunmaktadır. Veri bitleri ise 7 veya 8 bit olabilir.
  - > Parity bitinin gönderilmesi şart değildir.
- Asenkron iletişimde kullanılan diğer bir kavram ise veri gönderim hızıdır. Bu kavram "bps" (bits per second - saniyede gönderilen bit sayısı) birimi ile anılır. Standart veri gönderme hızları 110, 150, 300, 600, 1200, 2400, 4800, 9600, 56000, 115200 gibi hızdadır.
- Start biti "0" stop biti "1" verilerinden oluşmaktadır.

Asenkron Veri İletişim Diyagramı



# 09 I2C

5 Mayıs 2021 Çarşamba 08:03

## 09 I2C

### Giriş

- <https://www.ercankocclar.com/2018/01/i2c-iletisim-protokolu-ve-mikroc-kutuphanesi/> linkinden ayrıntılı bilgilere ulaşabiliriz.
- Pull-up direnci neden kullanıldığından bahsediyor.



- I2C protokolünün geliştirilme amacı, düşük hızlı çevre birimlerinin anakartları, cep telefonları, gömülü sistemler gibi elektronik cihazlara daha az kablo ihtiyacı ile bağlanabilmesini sağlamaktır.



- I2C iletişiminde sadece iki hat vardır. Bunlar SDA (Serial Data Line) ve SCL (Serial Clock Line) hatlarıdır. Bu hatlar ayrıca pull-up direncine ihtiyaç duyarlar.
- Genellikle +5V ve +3.3V voltajlarda çalışmakla beraber, I2C protokolü daha pratik voltaj seviyelerine de izin vermektedir.



- Yukarıda görüldüğü gibi I2C iletişimini aşağıdaki sıra ile gerçekleştirir:

  1. İlk olarak SDA ve SCL hatları HIGH (yüksek) konumdadırlar. Daha sonra SDA hattı master tarafından LOW(düşük) seviyeye çekilerek iletişimimin başlayacağı, slave cihazlara bildirilir. (Diyagramda S ile gösterilmiştir.)
  2. Bu bildirimi alan slave cihazlar, adres bilgisini beklemeye başlarlar. Adres bilgisi slave cihazların yapısına göre 7 bit, 10 bit veya 16 bit olabilirler. Master cihaz hangi slave cihaz ile haberleşmek istiyorsa onun adres bilgisini gönderdikten sonra, okuma mı yoksa yazma mı yapacağını belirtir. Adres hangi slave cihazın ise o cihaz master ile iletişim kurmaya başlar. Adres kendisine ait olan slave cihaz, master cihaza bir ACK (Acknowledge - Kabul) biti gönderir.
  3. Veri transferi işlemi gerçekleşir. Bu transfer iki yönlü de olabilir. (Slave'den Master'a veya Master'dan Slave'e.)

- I2C veriyolu multimeter (çoklu hükümeden) bir yapıdadır. Bu sayede iletişim hattında birden fazla cihaz olabilir. Master cihazlarda bir saat sinyali ve data gönderildiği anda diğer cihazların tamamı slave moduna geçerler.
- I2C düşük bant genişliğine sahiptir ve kısa mesafelerde kullanılır.
- SDA hattı haberleşmeyi başlatıp, sonlandırır. SCL ise veri hattı konfigurasyonunu sağlar.
- I2C hattı SDA hattının lojik high seviyesinden lojik low seviyeye düşmesi ile başlar. Aynı şekilde lojik low seviyesinden lojik high seviyeye çıkması ile sonlanır. SDA hattının haberleşmeyi başlatılabilmesi için SCL hattı da high olmalıdır.



- SCL hattı lojik high seviyesinde iken SDA hattı high seviyesine çekilirse haberleşme sonlanır.

- Multimaster I2C haberleşmesinde Repeated Start komutu vardır ve sıkılıkla kullanılır. I2C haberleşmesinde 2 adet cihaz olduğunu varsayılmı:
  - Birinci master cihaz start komutu gönderdi ve start komutundan sonra gerekli adres bilgilerini gönderdi. Tüm bu işlemler sürecinde I2C hattı birinci master cihaz tarafından kullanıldığından dolayı I2C hattı idle(boş) durumda olmayacağıdır. Birinci cihaz stop durumu göndermeden önce haberleşmede bir değişiklik yapmak isterse Repeated Start komutunu gönderir ve böylece 1.Master cihazın slave cihaz ile I2C haberleşmesi kopmamış olur. Multimaster olmayan durumlarda Repeated Start komutunu kullanmaya gerek yoktur. Repeated Start komutu ard arda gelen start stop komutlarından oluşur.



- I2C haberleşmesinde verinin gönderildiği veya verinin alındığını doğrulamak için ACK (Acknowledge) mesajları gönderilir. I2C haberleşmesinde 1 master cihaz ve birden fazla slave cihaz olduğunu varsayılmı. Master cihaz herhangi bir slave cihaza erişmek için start komutundan sonra ilgili slave cihazın adresini gönderir. Aynı hatta bağlı olan slave cihazların tamamı bu mesajları alır ancak sadece bu mesaja sahip olan slave cihaz Ack mesajını göndererek iletişim kurduğu master cihaza bildirir ve Ack mesajını alan master cihaz adres bilgisinden hemen sonra veri göndermeye başlar.
- I2C veri gönderimi:

|       |       |   |     |      |     |      |     |      |
|-------|-------|---|-----|------|-----|------|-----|------|
| Start | Adres | W | Ack | Data | Ack | Data | Ack | Stop |
|-------|-------|---|-----|------|-----|------|-----|------|

- Start biti "0" stop biti "1" verilerinden oluşmaktadır.
- Öncelikle master cihaz start komutunu gönderir ve devamında haberleşmek istediği slave cihazın 7 bitlik adresini ve devamında veri göndereceğini belirttiği W (write) komutunu gönderir. Bu mesajlardan sonra eğer slave adresi ile eşleşen bir slave cihaz var ise master cihaza Ack bilgisini gönderir.
- Master cihaz Ack bilgisini aldıktan sonra 8 bitlik detayı gönderir ve tekrar slave cihazdan veri alındığına dair

# 10 SPI

5 Mayıs 2021 Çarşamba 08:03

## 10 SPI

### Giriş

- SPI (Serial Peripheral Interface), STM32F4'ün desteklediği senkron seri haberleşme türlerinden biridir. Özellik ve kullanım olarak I2C'ye benzer. Bir STM32F4'ün diğer STM32F4 veya sensörlerle kısa mesafede haberleşmesini sağlar. SPI protokolünde de I2C'de olduğu gibi bir adet Master cihaz bulunur. Bu cihaz hatta bağlı çevresel cihazları kontrol eder.



- Çevresel cihazlarla veya diğer mikrodenetleyicilerle veri transferi sağlayan yazılım/donanım tabanlı seri iletişim protokolüdür. Bu haberleşme şekli karşılıklı iki tarafın clocklarının senkronize bir şekilde çalışmasıyla data iletişimini sağlamaktadır.
- Bunlar;
  - > MOSI (Master Output Slave Input): Master cihazdan Slave cihaza sinyal taşıyan hat
  - > MISO (Master Input Slave Output): Slave cihazdan Master cihaza sinyal taşıyan hat
- SCLK sinyali senkrol olarak bu iki MOSI ve MISO sinalının taşınmasında kullanılır. Bu sinyal sadece master cihaz tarafından üretilir.
- SPI'da veri transfer hızı I2C veri yolundan daha hızlıdır. Slave cihaz donanımsal olarak seçildiği için (SS veya CS pini üzerinden) slave cihaza I2C veri iletişimindeki gibi adres gönderilmez.
- Fakat birden fazla slave cihazın SPI veri yoluna bağlanması için birden fazla SS veya CS pini kullanılır.
- SPI iletişiminde, önce çalışmak istenilen slave cihazın bağlı olduğu SS pini seçilir.
- Master tarafından verinin en öncelikli kısmından (MSB) itibaren MOSI hattı üzerinden slave cihaz tarafına her clock pulsunda bir bit olmak üzere tüm veri gönderilir.



## 01 01 Harici Led Yakma

25 Aralık 2021 Cumartesi 00:52

01\_01 Harici Led Yakma

➤ HAL

## Konfigürasyon Kısmı

- Sistem saatini ayarlamak için öncelikle RCC kısmından HSE'den Crysal/Ceramic Resonator seçilir. Bu seçim işlemi ile beraber Clock Configuration da kutucuk bölme açılır.  
LSE'nin HSE ile aralarındaki farkı düşük hızlı olması ve düşük güç tüketimi için kullanılır.
  - System Clock Mux kısmında 3 seçeneğimiz var. Bunlar HSI, HSE ve PLLCLK'tur. HSI dahili iken HSE harici kaynaktır.

HSI secersek kutucuktaki değer olur, HSE secersek giriş frekansı belli MHz araliktadır. Biz burada kart üzerinde 8MHz olduğundan bu şekilde yazıp kullanıyoruz.

Eğer PLLCLK secersek ayarlamalarla maksimum MHz'e kadar istediğimiz değeri seçebiliriz.



- System Core kısmından SYS tıklıyoruz ve Debug'da Serial Wire diyoruz.

| Pin Name | Signal on Pin  | GPIO output | GPIO mode | GPIO Pull-up | Maximum ou... | User Label | Modified                 |
|----------|----------------|-------------|-----------|--------------|---------------|------------|--------------------------|
| PA13     | SYS_JTMS-SWDIO | n/a         | n/a       | n/a          | n/a           |            | <input type="checkbox"/> |
| PA14     | SYS_JTCK-SWCLK | n/a         | n/a       | n/a          | n/a           |            | <input type="checkbox"/> |

- RCC tıklıyoruz HSE'de Crystal/Ceramic Resonator işaretliyoruz.

| Pin Name    | Signal on Pin | GPIO output | GPIO mode | GPIO Pull-up | Maximum ou... | User Label | Modified                 |
|-------------|---------------|-------------|-----------|--------------|---------------|------------|--------------------------|
| PH0-OSC_IN  | RCC_OSC_IN    | n/a         | n/a       | n/a          | n/a           |            | <input type="checkbox"/> |
| PH1-OSC_OUT | RCC_OSC_OUT   | n/a         | n/a       | n/a          | n/a           |            | <input type="checkbox"/> |

- B portundaki 3., 4., 5. ve 6.pinlere led bağladık.

| Pin Name | Signal | GPIO o... | GPIO mode       | GPIO Pull-up/Pull-do...  | Maximum output | User Label | Modified                            |
|----------|--------|-----------|-----------------|--------------------------|----------------|------------|-------------------------------------|
| PB3      | n/a    | Low       | Output Push ... | No pull-up and no pul... | Very High      | LED1       | <input checked="" type="checkbox"/> |
| PB4      | n/a    | Low       | Output Push ... | No pull-up and no pul... | Very High      | LED2       | <input checked="" type="checkbox"/> |
| PB5      | n/a    | Low       | Output Push ... | No pull-up and no pul... | Very High      | LED3       | <input checked="" type="checkbox"/> |
| PB6      | n/a    | Low       | Output Push ... | No pull-up and no pul... | Very High      | LED4       | <input checked="" type="checkbox"/> |

### Kod Kısımları

- CubeMX'de ayarlamaları yaptıktan sonra main.c kısmında yaptığımız kısımları kendisi kod içerisinde oluşturmuştur.
- RCC ve GPIO için iki ayrı fonksiyon oluşturmuştur.
- Oluşturulan iki fonksiyon kod içerisinde yazmadan önce private function prototypes kısmında belirtilir.

```

48 /* Private function prototypes -----*/
49 void SystemClock_Config(void);
50 static void MX_GPIO_Init(void);
51
52 int main(void)
53 {
54     /* USER CODE BEGIN 1 */
55
56     /* USER CODE END 1 */
57
58     /* MCU Configuration-----*/
59
60     /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
61     HAL_Init();
62
63     /* USER CODE BEGIN Init */
64
65     /* USER CODE END Init */
66
67     /* Configure the system clock */
68     SystemClock_Config();
69
70     /* USER CODE BEGIN SysInit */
71
72     /* Initialize all configured peripherals */
73     MX_GPIO_Init();
74
75     /* USER CODE BEGIN 2 */
76
77     /* Infinite loop */
78     /* USER CODE BEGIN WHILE */
79     while (1)
80     {
81         /* USER CODE END WHILE */
82
83         /* USER CODE BEGIN 3 */
84     }
85
86     /* USER CODE END 3 */
87 }
```

- hal\_gpio.c kısmından WritePin ve Toggle Pin fonksiyonlarına ulaşabiliriz.

```

410 void HAL_GPIO_WritePin(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin, GPIO_PinState PinState)
411 {
412     /* Check the parameters */
413     assert_param(IS_GPIO_PIN(GPIO_Pin));
414     assert_param(IS_GPIO_PIN_ACTION(PinState));
415
416     if(PinState != GPIO_PIN_RESET)
417     {
418         GPIOx->BSRR = GPIO_Pin;
419     }
420     else
421     {
422         GPIOx->BSRR = (uint32_t)GPIO_Pin << 16U;
423     }
424 }
```

```

433 void HAL_GPIO_TogglePin(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin)
434 {
435     uint32_t odr;
436
437     /* Check the parameters */
438     assert_param(IS_GPIO_PIN(GPIO_Pin));
439
440     /* get current Output Data Register value */
441     odr = GPIOx->ODR;
442
443     /* Set selected pins that were at low level, and reset ones that were high */
444     GPIOx->BSRR = ((odr & GPIO_Pin) << GPIO_NUMBER) | (~odr & GPIO_Pin);
445 }

```

- Döngümüze aşağıdaki kodu yazarız.

```

94     while (1)
95     {
96         /* USER CODE END WHILE */
97
98         /* USER CODE BEGIN 3 */
99         HAL_GPIO_WritePin(GPIOB, LED1_Pin | LED2_Pin | LED3_Pin | LED4_Pin, GPIO_PIN_SET);
100        HAL_Delay(2000);
101        HAL_GPIO_TogglePin(GPIOB, LED1_Pin | LED2_Pin | LED3_Pin | LED4_Pin);
102        HAL_Delay(500);
103    }

```

## ➤ REGISTER

### Konfigürasyon Kısımlı

- main.c dosyasındaki yorum satırları silip sadece hale getiriyoruz.

```

1 #include "stm32f4xx.h"
2
3 int main(void)
4 {
5     while (1)
6     {
7
8     }
9 }

• system_stmf4xx.c dosyasındaki 182.satırındaki SystemCoreClock kısmına Ctrl ile sağ tıklıyoruz ve bizi
system_stmf4xx.h dosyasındaki 59.satırına götürüyor. Bu satırı kopyalayıp main.c dosyasına yapıştırıyoruz.

182     uint32_t SystemCoreClock = 168000000;

```

```

59 extern uint32_t SystemCoreClock;           /*!< System Clock Frequency (Core Clock) */

1 #include "stm32f4xx.h"
2
3 extern uint32_t SystemCoreClock;
4
5 uint32_t systemClock;
6
7 int main(void)
8 {
9     systemClock=SystemCoreClock;
10
11     while (1)
12     {
13
14     }
15 }

```

| Expression      | Type     | Value     |
|-----------------|----------|-----------|
| 00= systemClock | uint32_t | 168000000 |

```

1 #include "stm32f4xx.h"
2
3 extern uint32_t SystemCoreClock;
4
5 uint32_t systemClock;
6
7 int main(void)
8 {
9     systemClock=SystemCoreClock;          //168 000 000
10
11     RCC_DeInit();                      //HSI ON PLL OFF
12
13     SystemCoreClockUpdate();
14     systemClock=SystemCoreClock;        //16 000 000
15
16     while (1)
17     {
18
19     }
20 }

```

| Expression               | Type     | Value    |
|--------------------------|----------|----------|
| <code>systemClock</code> | uint32_t | 16000000 |

- STM32F407VG mikrodenetleyicinin Reference Manuals'de RCC kısmına bakıyoruz. Mikrodenetleyici 32 bittir. Bu sebeple kaydediciler olan registerler 32 bittir.
- Bit değerlerinde yazan r ile rw'nin anlamı r'nin read yani okunabilir, w'nin write yani yazılabılır anlamı vardır. r olan yerlere müdahale edemiyoruz ama rw yazan bitlere müdahale edebiliyoruz.

### RCC clock control register (RCC\_CR)

Address offset: 0x00

Reset value: 0x0000 XX83 where X is undefined.

Access: no wait state, word, half-word and byte access

|    |    |    |    |           |          |          |       |    |    |    |    |        |         |         |        |
|----|----|----|----|-----------|----------|----------|-------|----|----|----|----|--------|---------|---------|--------|
| 31 | 30 | 29 | 28 | 27        | 26       | 25       | 24    | 23 | 22 | 21 | 20 | 19     | 18      | 17      | 16     |
|    |    |    |    | PLL2S RDY | PLL2S ON | PLL RD Y | PLLON |    |    |    |    | CSS ON | HSE BYP | HSE RDY | HSE ON |
|    |    |    |    | r         | rw       | r        | rw    |    |    |    |    | rw     | rw      | r       | rw     |
| 15 | 14 | 13 | 12 | 11        | 10       | 9        | 8     | 7  | 6  | 5  | 4  | 3      | 2       | 1       | 0      |
|    |    |    |    |           |          |          |       |    |    |    |    |        |         | HSI RDY | HSION  |
| r  | r  | r  | r  | r         | r        | r        | r     | rw | rw | rw | rw | rw     | rw      | r       | rw     |

- Adress offset bizim ilk olarak kaydedilen yeri gösterir.
- Reset value ise girilen değer sonrası tüm bitler sıfırlanır.
- Burada resetleme işlemi yapıyoruz.

`RCC->CR &= 0x00000083;`

- "&=" anlam kaydet anlamındadır.
- Harici osilatör kullanacağımızdan HSEON olan 16.biti 1 yapmam gerekiyor.

Bit 16 **HSEON**: HSE clock enable

Set and cleared by software.

Cleared by hardware to stop the HSE oscillator when entering Stop or Standby mode. This bit cannot be reset if the HSE oscillator is used directly or indirectly as the system clock.

0: HSE oscillator OFF

1: HSE oscillator ON

- "|=" anlamı ise öncekine 1 ekle ve eşitle anlamındadır. Bunu kullanırken ekleme yani öteleme yapıyoruz.
- 16.biti 1 yapmak için öteleme kullanırken  $1 \ll 16$  yazıyoruz yani 0.bitte 1 yapıp 16 bit öteliyor.

`RCC->CR |= 1 << 16;`

Bunu denetlemek için 17.biti kullanıyoruz. Yani 16.bit 1 olup olmadığı HSE osilatör 6 clock cycles yaptıktan sonra 17.bit olan HSERDY biti 1 oluyor.

Bit 17 **HSERDY**: HSE clock ready flag

Set by hardware to indicate that the HSE oscillator is stable. After the HSEON bit is cleared, HSERDY goes low after 6 HSE oscillator clock cycles.

0: HSE oscillator not ready

1: HSE oscillator ready

- While döngüsü 1 olduğu sürece çalışır. HSERDY biti 1 olana dek 0 olacağından ve döngünün çalışabilmesi için başına "!" işaretini koyarak tersliyoruz.
- 17.bit 1 olduğunda döngüden çıkışacaktır.
- & biti lojik kapılarda olduğu gibi 0&0 ile 0&1 olduğunda çıkışında 0 verir. 1&1 olduğunda çıkışında 1 verir.
- Buradaki  $1 \ll 17$  işlemini yapabilmesi için 6 clock cycles zaman geçmesi gerekiyor. Öteleme işlem yaptığını anlamak için RCC->CR 1 olduğunda 17.bit 1 olmuş oluyor. Böylece 1&1'den 1 oluyor ve !1'den 0 olarak döngüden çıkışıyor.

`while(!(RCC->CR & (1 << 17)));`

- HSI kapatmak için HSION bitini 0 yapmam gerekiyor.

Bit 0 **HSION**: Internal high-speed clock enable

Set and cleared by software.

Set by hardware to force the HSI oscillator ON when leaving the Stop or Standby mode or in case of a failure of the HSE oscillator used directly or indirectly as the system clock. This bit cannot be cleared if the HSI is used directly or indirectly as the system clock.

0: HSI oscillator OFF

1: HSI oscillator ON

- $1 \ll 0$  ile 0.bit 1 yapılır ardından ~ işaretini ile 0 yapılır.

`RCC->CR &= ~(1 << 0);`

- Şu an mikrodenetleyici 8 000 000Hz'de çalışıyor. PLL ile 168 000 000Hz'e çıkaracağız.

Bit 19 **CSSON**: Clock security system enable

Set and cleared by software to enable the clock security system. When CSSON is set, the clock detector is enabled by hardware when the HSE oscillator is ready, and disabled by hardware if an oscillator failure is detected.

0: Clock security system OFF (Clock detector OFF)

1: Clock security system ON (Clock detector ON if HSE oscillator is stable, OFF if not)

- 19.bit 1 yapılır.

RCC->**CR** |= 1 << 19;

PLLON 24.bit 1 yapılmadan önce PLL'nin konfigürasyonlarını yapmamız gerekiyor.

### RCC PLL configuration register (RCC\_PLLCFGR)

Address offset: 0x04

Reset value: 0x2400 3010

Access: no wait state, word, half-word and byte access.

This register is used to configure the PLL clock outputs according to the formulas:

- $f_{(VCO\ clock)} = f_{(PLL\ clock\ input)} \times (PLLQN / PLLM)$
- $f_{(PLL\ general\ clock\ output)} = f_{(VCO\ clock)} / PLLP$
- $f_{(USB\ OTG\ FS,\ SDIO,\ RNG\ clock\ output)} = f_{(VCO\ clock)} / PLLQ$

|           |    |    |       |       |       |       |           |        |    |          |       |       |       |       |       |       |
|-----------|----|----|-------|-------|-------|-------|-----------|--------|----|----------|-------|-------|-------|-------|-------|-------|
| 31        | 30 | 29 | 28    | 27    | 26    | 25    | 24        | 23     | 22 | 21       | 20    | 19    | 18    | 17    | 16    |       |
| Reserved  |    |    | PLLQ3 | PLLQ2 | PLLQ1 | PLLQ0 | Reserv ed | PLLSRC |    | Reserved |       |       | PLLPI | PLLPO |       |       |
| 15        | 14 | 13 | 12    | 11    | 10    | 9     | 8         | rw     | 6  | 5        | 4     | 3     | 2     | 1     |       |       |
| Reserv ed |    |    |       |       |       |       |           |        |    | PLLN     | PLLM5 | PLLM4 | PLLM3 | PLLM2 | PLLM1 | PLLM0 |
|           | rw | rw | rw    | rw    | rw    | rw    | rw        | rw     | rw | rw       | rw    | rw    | rw    | rw    | rw    | rw    |

- PLL için M, N ve P değerlerine göre sistem saatı 168MHz yapıyoruz. Bunun için M değerine 4, N değerine 168 ve P değerine 2 verdigimizde bu işlemi sağlamış oluyoruz.
- M için PLLM kısmında ilk 6 bitte yazacağz. Bunun için 000100 değeri için 0., 1., 3., 4. ve 5.bite 0 yapıyorken 2.biti 1 yapıyoruz.
- 2.biti 1 yaparken ekleme yaparak yani "|=" işaretini kullanırken diğerlerine sadece 0 bite kaydetmek için "&=" işaretini kullanıyoruz.

Bits 5:0 **PLLM**: Division factor for the main PLL (PLL) and audio PLL (PLLI2S) input clock

Set and cleared by software to divide the PLL and PLLI2S input clock before the VCO.

These bits can be written only when the PLL and PLLI2S are disabled.

**Caution:** The software has to set these bits correctly to ensure that the VCO input frequency ranges from 1 to 2 MHz. It is recommended to select a frequency of 2 MHz to limit PLL jitter.

VCO input frequency = PLL input clock frequency / PLLM with  $2 \leq PLLM \leq 3$

00000: PLLM = 0, wrong configuration

00001: PLLM = 1, wrong configuration

00010: PLLM = 2

00011: PLLM = 3

00100: PLLM = 4

...

111110: PLLM = 62

111111: PLLM = 63

```
RCC->PLLCFGR &= ~(1 << 0);           //PLLM0 0
RCC->PLLCFGR &= ~(1 << 1);           //PLLM1 0
RCC->PLLCFGR |= (1 << 2);            //PLLM2 1
RCC->PLLCFGR &= ~(1 << 3);            //PLLM3 0
RCC->PLLCFGR &= ~(1 << 4);            //PLLM4 0
RCC->PLLCFGR &= ~(1 << 5);            //PLLM5 0
```

- N için PLLN kısmını kullanıyoruz.

Bits 14:6 **PLLN**: Main PLL (PLL) multiplication factor for VCO

Set and cleared by software to control the multiplication factor of the VCO. These bits can be written only when PLL is disabled. Only half-word and word accesses are allowed to write these bits.

**Caution:** The software has to set these bits correctly to ensure that the VCO output frequency is between 100 and 432 MHz.

VCO output frequency = VCO input frequency  $\times$  PLLN with  $50 \leq PLLN \leq 432$

00000000: PLLN = 0, wrong configuration

00000001: PLLN = 1, wrong configuration

...

000110010: PLLN = 50

...

001100011: PLLN = 99

001100100: PLLN = 100

...

110110000: PLLN = 432

110110001: PLLN = 433, wrong configuration

...

111111111: PLLN = 511, wrong configuration

**Note:** Multiplication factors ranging from 50 and 99 are possible for VCO input frequency higher than 1 MHz. However care must be taken that the minimum VCO output frequency respects the value specified above.

|     |     |
|-----|-----|
| HEX | A8  |
| DEC | 168 |
| OCT | 256 |

|     |           |
|-----|-----------|
| HEX | A8        |
| DEC | 168       |
| OCT | 250       |
| BIN | 1010 1000 |

- 168 decimal sayısının binary karşılığı 1010 1000'dir. Biz burada 9 bit olduğundan 010101000 yazacağız.  
Biz bununla uğraşmak yerine 6 bit öteleyip 168 sayısını yazacağız.

RCC->PLLCFGR |= (168 << 6); //PLLN 168

Aynı işlemi M için de yapabiliriz.

RCC->PLLCFGR |= (4 << 0); //PLLP 4

P için PLLP kısmını kullanıyoruz. 2 değeri için iki biti 0 yap diyor.

Bits 17:16 PLLP: Main PLL (PLL) division factor for main system clock

Set and cleared by software to control the frequency of the general PLL output clock. These bits can be written only if PLL is disabled.

**Caution:** The software has to set these bits correctly not to exceed 168 MHz on this domain.

PLL output clock frequency = VCO frequency / PLLP with PLLP = 2, 4, 6, or 8

00: PLLP = 2

01: PLLP = 4

10: PLLP = 6

11: PLLP = 8

RCC->PLLCFGR &= ~(1 << 16); //PLLP1 0

RCC->PLLCFGR &= ~(1 << 17); //PLLP2 0

- PLL için başlangıçta tüm bitleri 0 yapıyoruz.

- Burada 8 tane sıfır var. Buradaki her biri 32 bit'de 4 bite karşılık geliyor.

RCC->CFGR = 0x00000000;

- Tüm bitleri başlangıçta 0 yaptığımız için PLL'de P için yaptığımız 0'lama işlemine gerek kalmadı.

```
7 void RCC_Config(void)
8 {
9     //RCC->CR &= 0x00000083;           //RESET
10
11    RCC->CR &= ~(1 << 0);          //HSION
12    RCC->CR |= 1 << 16;            //HSEON
13    while(!(RCC->CR & (1 << 17))); //HSERDY
14    RCC->CR |= 1 << 19;            //CSSON
15    RCC->CFGR = 0x00000000;
16    //RCC->PLLCFGR &= ~(1 << 0);    //PLLM0 0
17    //RCC->PLLCFGR &= ~(1 << 1);    //PLLM1 0
18    //RCC->PLLCFGR |= (1 << 2);     //PLLM2 1
19    //RCC->PLLCFGR &= ~(1 << 3);    //PLLM3 0
20    //RCC->PLLCFGR &= ~(1 << 4);    //PLLM4 0
21    //RCC->PLLCFGR &= ~(1 << 5);    //PLLM5 0
22    RCC->PLLCFGR |= (4 << 0);      //PLLM 4
23    RCC->PLLCFGR |= (168 << 6);    //PLLN 168
24    //RCC->PLLCFGR &= ~(1 << 16);   //PLLP1 0
25    //RCC->PLLCFGR &= ~(1 << 17);   //PLLP2 0
26 }
```

| SFRs Registers   |            |         |
|------------------|------------|---------|
| type filter text |            |         |
| Register         | Address    | Value   |
| > DMA1           |            |         |
| ▼ RCC            |            |         |
| > CR             | 0x40023800 | 0xb7083 |
| ▼ PLLCFGR        | 0x40023804 | 0x2a04  |
| ■■■■■ PLLQ3      | [27:1]     | 0x0     |
| ■■■■■ PLLQ2      | [26:1]     | 0x0     |
| ■■■■■ PLLQ1      | [25:1]     | 0x0     |
| ■■■■■ PLLQ0      | [24:1]     | 0x0     |
| ■■■■■ PLLSRC     | [22:1]     | 0x0     |
| ■■■■■ PLLP1      | [17:1]     | 0x0     |
| ■■■■■ PLLP0      | [16:1]     | 0x0     |
| ■■■■■ PLLN8      | [14:1]     | 0x0     |
| ■■■■■ PLLN7      | [13:1]     | 0x1     |
| ■■■■■ PLLN6      | [12:1]     | 0x0     |
| ■■■■■ PLLN5      | [11:1]     | 0x1     |
| ■■■■■ PLLN4      | [10:1]     | 0x0     |
| ■■■■■ PLLN3      | [9:1]      | 0x1     |
| ■■■■■ PLLN2      | [8:1]      | 0x0     |
| ■■■■■ PLLN1      | [7:1]      | 0x0     |
| ■■■■■ PLLN0      | [6:1]      | 0x0     |
| ■■■■■ PLLM5      | [5:1]      | 0x0     |
| ■■■■■ PLLM4      | [4:1]      | 0x0     |
| ■■■■■ PLLM3      | [3:1]      | 0x0     |
| ■■■■■ PLLM2      | [2:1]      | 0x1     |
| ■■■■■ PLLM1      | [1:1]      | 0x0     |
| ■■■■■ PLLM0      | [0:1]      | 0x0     |

- PLLSRC ile PLL sürücüsü seçilir. Bu bit için HSI kullanırsak 0, HSE kullanırsak 1 yapılır. Biz HSE kullandığımızdan bu biti 1 yapıyoruz.

Bit 22 PLLSRC: Main PLL(PLL) and audio PLL (PLLI2S) entry clock source

Set and cleared by software to select PLL and PLLI2S clock source. This bit can be written only when PLL and PLLI2S are disabled.

Bit 22 **PLLSRC**: Main PLL(PLL) and audio PLL (PLLI2S) entry clock source  
Set and cleared by software to select PLL and PLLI2S clock source. This bit can be written only when PLL and PLLI2S are disabled.  
0: HSI clock selected as PLL and PLLI2S clock entry  
1: HSE oscillator clock selected as PLL and PLLI2S clock entry  
RCC->**CFGREG** |= (1 << 22);

Bit 24 **PLLON**: Main PLL (PLL) enable  
Set and cleared by software to enable PLL.  
Cleared by hardware when entering Stop or Standby mode. This bit cannot be reset if PLL clock is used as the system clock.  
0: PLL OFF  
1: PLL ON  
RCC->**CR** |= 1 << 24;

Bit 25 **PLLRDY**: Main PLL (PLL) clock ready flag  
Set by hardware to indicate that PLL is locked.  
0: PLL unlocked  
1: PLL locked  
**while**(!(RCC->**CR** & (1 << 25)));

### RCC clock configuration register (RCC\_CFGREG)

Address offset: 0x08

Reset value: 0x0000 0000

Access: 0 ≤ wait state ≤ 2, word, half-word and byte access

1 or 2 wait states inserted only if the access occurs during a clock source switch.

| 31         | 30            | 29         | 28 | 27 | 26            | 25       | 24        | 23     | 22   | 21 | 20          | 19   | 18  | 17  | 16 |
|------------|---------------|------------|----|----|---------------|----------|-----------|--------|------|----|-------------|------|-----|-----|----|
| MCO2       | MCO2 PRE[2:0] |            |    |    | MCO1 PRE[2:0] |          |           | I2SSCR | MCO1 |    | RTCPRE[4:0] |      |     |     |    |
| rw         | rw            | rw         | rw | rw | rw            | rw       | rw        | rw     | rw   | rw | rw          | rw   | rw  | rw  | rw |
| 15         | 14            | 13         | 12 | 11 | 10            | 9        | 8         | 7      | 6    | 5  | 4           | 3    | 2   | 1   | 0  |
| PPRE2[2:0] |               | PPRE1[2:0] |    |    |               | Reserved | HPRE[3:0] |        |      |    | SWS1        | SWS0 | SW1 | SW0 |    |
| rw         | rw            | rw         | rw | rw | rw            |          | rw        | rw     | rw   | rw | r           | r    | rw  | rw  |    |

- İlk önce 0.bit 0 yapıldı daha sonra 1.bit 1 yapılarak sistemin saatini PLL ile ayarladığımızı belirtiyoruz.

Bits 1:0 **SW**: System clock switch

Set and cleared by software to select the system clock source.  
Set by hardware to force the HSI selection when leaving the Stop or Standby mode or in case of failure of the HSE oscillator used directly or indirectly as the system clock.  
00: HSI oscillator selected as system clock  
01: HSE oscillator selected as system clock  
10: PLL selected as system clock  
11: not allowed

RCC->**CFGREG** &= ~(1 << 0);

RCC->**CFGREG** |= (1 << 1);

Bits 3:2 **SWS**: System clock switch status

Set and cleared by hardware to indicate which clock source is used as the system clock.  
00: HSI oscillator used as the system clock  
01: HSE oscillator used as the system clock  
10: PLL used as the system clock  
11: not applicable

**while**(!(RCC->**CR** & (1 << 1)));

```

7 void RCC_Config(void)
8 {
9     //RCC->CR &= 0x00000083;           //RESET
10
11    RCC->CR &= ~(1 << 0);          //HSION
12    RCC->CR |= 1 << 16;            //HSEON
13    while(!(RCC->CR & (1 << 17))); //HSERDY
14    RCC->CR |= 1 << 19;            //CSSON
15    RCC->CFGR = 0x00000000;
16    RCC->PLLCFGR |= (1 << 22);    //PLLSRC
17    //RCC->PLLCFGR &= ~(1 << 0);   //PLLM0 0
18    //RCC->PLLCFGR &= ~(1 << 1);   //PLLM1 0
19    //RCC->PLLCFGR |= (1 << 2);    //PLLM2 1
20    //RCC->PLLCFGR &= ~(1 << 3);    //PLLM3 0
21    //RCC->PLLCFGR &= ~(1 << 4);    //PLLM4 0
22    //RCC->PLLCFGR &= ~(1 << 5);    //PLLM5 0
23    RCC->PLLCFGR |= (4 << 0);      //PLLM 4
24    RCC->PLLCFGR |= (168 << 6);    //PLLN 168
25    //RCC->PLLCFGR &= ~(1 << 16);   //PLLP1 0
26    //RCC->PLLCFGR &= ~(1 << 17);   //PLLP2 0
27
28    RCC->CR |= 1 << 24;            //PLLON
29    while(!(RCC->CR & (1 << 25))); //PLLRDY
30
31    RCC->CFGR &= ~(1 << 0);
32    RCC->CFGR |= (1 << 1);        //SW
33
34    while(!(RCC->CR & (1 << 1))); //SWS
35 }
36 int main(void)
37 {
38     //systemClock=SystemCoreClock;    //168 000 000
39
40     //RCC_DeInit();                 //HSI ON PLL OFF
41
42     //SystemCoreClockUpdate();
43     //systemClock=SystemCoreClock;    //16 000 000
44
45     RCC_Config();
46     SystemCoreClockUpdate();
47     systemClock=SystemCoreClock;    //168 000 000
48
49     while (1)
50     {
51
52     }
53 }
54 }
```

| Expression       | Type     | Value     |
|------------------|----------|-----------|
| (0): systemClock | uint32_t | 168000000 |

### GPIO port mode register (GPIOx\_MODER) (x = A..I/J/K)

Address offset: 0x00

Reset values:

- 0xA800 0000 for port A
- 0x0000 0280 for port B
- 0x0000 0000 for other ports

| 31           | 30 | 29           | 28 | 27           | 26 | 25           | 24 | 23           | 22 | 21           | 20 | 19          | 18 | 17          | 16 |
|--------------|----|--------------|----|--------------|----|--------------|----|--------------|----|--------------|----|-------------|----|-------------|----|
| MODER15[1:0] |    | MODER14[1:0] |    | MODER13[1:0] |    | MODER12[1:0] |    | MODER11[1:0] |    | MODER10[1:0] |    | MODER9[1:0] |    | MODER8[1:0] |    |
| rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw          | rw | rw          | rw |
| 15           | 14 | 13           | 12 | 11           | 10 | 9            | 8  | 7            | 6  | 5            | 4  | 3           | 2  | 1           | 0  |
| MODER7[1:0]  |    | MODER6[1:0]  |    | MODER5[1:0]  |    | MODER4[1:0]  |    | MODER3[1:0]  |    | MODER2[1:0]  |    | MODER1[1:0] |    | MODERO[1:0] |    |
| rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw          | rw | rw          | rw |

- Pinlerin nasıl kullanacağımızı seçtiğimiz kısımdır. Her iki bit bir pini temsil ediyor.
- Pini giriş, çıkış, analog ya da alternatif fonksiyon olarak mı kullanacağımızı belirtiyoruz. Alternatif fonksiyon ile kast edilen pinin çevresel birimlerden I2C, SPI olarak kullanılacağını belirtiyor.

Bits 2y:2y+1 MODER[y:0]: Port x configuration bits (y = 0..15)

These bits are written by software to configure the I/O direction mode.

00: Input (reset state)

01: General purpose output mode

10: Alternate function mode

11: Analog mode

- STM32f407VG mikrodenetleyicisinin board üzerindeki led pinleri D12, D13, D14 ve D15'tir.



## LEDs

- D portundaki belirlediğimiz pinlerimizi output yani çıkış olarak tanımlıyoruz ama öncesinde D portunu clock hattına aktarmamız gerekiyor.

```

GPIOD->MODER |= 1 << 24;           //PD12
GPIOD->MODER &= ~(1 << 25);
GPIOD->MODER |= 1 << 26;           //PD13
GPIOD->MODER &= ~(1 << 27);
GPIOD->MODER |= 1 << 28;           //PD14
GPIOD->MODER &= ~(1 << 29);
GPIOD->MODER |= 1 << 30;           //PD15
GPIOD->MODER &= ~(1 << 31);

```

- Output ayarı için her bir pine bir bit ayrılmıştır. Kullanacağımız push-pull reset durumunda 0 olduğundan bir ayar yapmamıza gerek yok.

### GPIO port output type register (GPIOx\_OTYPER) (x = A..I/J/K)

Address offset: 0x04

Reset value: 0x0000 0000

| 31       | 30   | 29   | 28   | 27   | 26   | 25  | 24  | 23  | 22  | 21  | 20  | 19  | 18  | 17  | 16  |
|----------|------|------|------|------|------|-----|-----|-----|-----|-----|-----|-----|-----|-----|-----|
| Reserved |      |      |      |      |      |     |     |     |     |     |     |     |     |     |     |
| 15       | 14   | 13   | 12   | 11   | 10   | 9   | 8   | 7   | 6   | 5   | 4   | 3   | 2   | 1   | 0   |
| OT15     | OT14 | OT13 | OT12 | OT11 | OT10 | OT9 | OT8 | OT7 | OT6 | OT5 | OT4 | OT3 | OT2 | OT1 | OT0 |

Bits 31:16 Reserved, must be kept at reset value.

Bits 15:0 OTy: Port x configuration bits (y = 0..15)

These bits are written by software to configure the output type of the I/O port.

0: Output push-pull (reset state)

1: Output open-drain

- Pinlerin very high speed olmasını istiyoruz. Pinleri 1 yapıyoruz.

### GPIO port output speed register (GPIOx\_OSPEEDR) (x = A..I/J/K)

Address offset: 0x08

Reset values:

- 0x0C00 0000 for port A
- 0x0000 00C0 for port B
- 0x0000 0000 for other ports

| 31              | 30 | 29              | 28 | 27              | 26 | 25              | 24 | 23              | 22 | 21              | 20 | 19             | 18 | 17             | 16 |
|-----------------|----|-----------------|----|-----------------|----|-----------------|----|-----------------|----|-----------------|----|----------------|----|----------------|----|
| OSPEEDR15 [1:0] |    | OSPEEDR14 [1:0] |    | OSPEEDR13 [1:0] |    | OSPEEDR12 [1:0] |    | OSPEEDR11 [1:0] |    | OSPEEDR10 [1:0] |    | OSPEEDR9 [1:0] |    | OSPEEDR8 [1:0] |    |
| rw              | rw | rw              | rw | rw              | rw | rw              | rw | rw              | rw | rw              | rw | rw             | rw | rw             | rw |
| 15              | 14 | 13              | 12 | 11              | 10 | 9               | 8  | 7               | 6  | 5               | 4  | 3              | 2  | 1              | 0  |
| OSPEEDR7[1:0]   |    | OSPEEDR6[1:0]   |    | OSPEEDR5[1:0]   |    | OSPEEDR4[1:0]   |    | OSPEEDR3[1:0]   |    | OSPEEDR2[1:0]   |    | OSPEEDR1 [1:0] |    | OSPEEDR0 [1:0] |    |
| rw              | rw | rw              | rw | rw              | rw | rw              | rw | rw              | rw | rw              | rw | rw             | rw | rw             | rw |

Bits 2y:2y+1 OSPEEDRy[1:0]: Port x configuration bits (y = 0..15)

These bits are written by software to configure the I/O output speed.

00: Low speed

01: Medium speed

10: High speed

11: Very high speed

Note: Refer to the product datasheets for the values of OSPEEDRy bits versus V<sub>DD</sub> range and external load.

```

GPIOD->OSPEEDR |= 1 << 24;
GPIOD->OSPEEDR |= 1 << 25;
GPIOD->OSPEEDR |= 1 << 26;
GPIOD->OSPEEDR |= 1 << 27;
GPIOD->OSPEEDR |= 1 << 28;
GPIOD->OSPEEDR |= 1 << 29;
GPIOD->OSPEEDR |= 1 << 30;
GPIOD->OSPEEDR |= 1 << 31;

```

- Bu şekilde tek tek yazmak yerine 32 bitinin son sekize 1 yazarak hex kısmı yazabiliriz.

| HEX                        | FF00 0000                               |
|----------------------------|-----------------------------------------|
| DEC                        | -16.777.216                             |
| OCT                        | 37 700 000 000                          |
| BIN                        | 1111 1111 0000 0000 0000 0000 0000 0000 |
|                            | WORD MS M <sup>+</sup>                  |
| 0000 0000 0000 0000        | 60 56 52 48                             |
| 0000 0000 0000 0000        | 44 40 36 32                             |
| <b>1111 1111 0000 0000</b> | 28 24 20 16                             |
| 0000 0000 0000 0000        | 12 8 4 0                                |

```
GPIOD->OSPEEDR |= 0xFF000000;
```

- 8 biti 1 yaptığımızda HEX olarak FF veriri. 24.bite eklemeli şekilde de yapabiliriz.

```
GPIOD->OSPEEDR |= FF << 24;
```

- Kullanaçagımız no pull-up, no pull-down reset durumunda 0 olduğundan bir ayar yapmamıza gerek yok.

#### GPIO port pull-up/pull-down register (GPIOx\_PUPDR)

(x = A..I/J/K)

Address offset: 0x0C

Reset values:

- 0x6400 0000 for port A
- 0x0000 0100 for port B
- 0x0000 0000 for other ports

| 31           | 30           | 29           | 28           | 27           | 26           | 25          | 24          | 23 | 22 | 21 | 20 | 19 | 18 | 17 | 16 |
|--------------|--------------|--------------|--------------|--------------|--------------|-------------|-------------|----|----|----|----|----|----|----|----|
| PUPDR15[1:0] | PUPDR14[1:0] | PUPDR13[1:0] | PUPDR12[1:0] | PUPDR11[1:0] | PUPDR10[1:0] | PUPDR9[1:0] | PUPDR8[1:0] |    |    |    |    |    |    |    |    |
| rw           | rw           | rw           | rw           | rw           | rw           | rw          | rw          | rw | rw | rw | rw | rw | rw | rw | rw |
| 15           | 14           | 13           | 12           | 11           | 10           | 9           | 8           | 7  | 6  | 5  | 4  | 3  | 2  | 1  | 0  |
| PUPDR7[1:0]  | PUPDR6[1:0]  | PUPDR5[1:0]  | PUPDR4[1:0]  | PUPDR3[1:0]  | PUPDR2[1:0]  | PUPDR1[1:0] | PUPDR0[1:0] |    |    |    |    |    |    |    |    |
| rw           | rw           | rw           | rw           | rw           | rw           | rw          | rw          | rw | rw | rw | rw | rw | rw | rw | rw |

Bits 2y:2y+1 PUPDR<sub>y</sub>[1:0]: Port x configuration bits (y = 0..15)

These bits are written by software to configure the I/O pull-up or pull-down

00: No pull-up, pull-down

01: Pull-up

10: Pull-down

11: Reserved

- Öncesinde clock hattını aktif etmemiz gerekiyor. Portlar AHB1 kısmına gidiyor.





### RCC AHB1 peripheral clock enable register (RCC\_AHB1ENR)

Address offset: 0x30

Reset value: 0x0010 0000

Access: no wait state, word, half-word and byte access.

| 31            | 30        | 29          | 28                  | 27                 | 26                 | 25               | 24          | 23          | 22         | 21               | 20          | 19            | 18          | 17          | 16       |  |
|---------------|-----------|-------------|---------------------|--------------------|--------------------|------------------|-------------|-------------|------------|------------------|-------------|---------------|-------------|-------------|----------|--|
| Reser-<br>ved | OTGH<br>S | OTGH<br>SEN | ETHM<br>ACPTP<br>EN | ETHM<br>ACRXE<br>N | ETHM<br>ACTXE<br>N | ETHM<br>A<br>CEN | Reserved    | DMA2E<br>N  | DMA1E<br>N | CCMDAT<br>ARAMEN | Res.        | BKPSR<br>AMEN | Reserved    | Reserved    | Reserved |  |
|               | rw        | rw          | rw                  | rw                 | rw                 | rw               |             | rw          | rw         |                  |             | rw            |             |             |          |  |
| 15            | 14        | 13          | 12                  | 11                 | 10                 | 9                | 8           | 7           | 6          | 5                | 4           | 3             | 2           | 1           | 0        |  |
| Reserved      |           | CRCE<br>N   | Reserved            |                    |                    | GPIOIE<br>N      | GPIOH<br>EN | GPIOG<br>EN | GPIOF<br>N | GPIOEEN          | GPIOD<br>EN | GPIOC<br>EN   | GPIO<br>BEN | GPIO<br>AEN |          |  |
|               |           | rw          |                     |                    |                    | rw               | rw          | rw          | rw         | rw               | rw          | rw            | rw          | rw          |          |  |

- Biz D portunu kullandığımızdan sadece bunu aktif ediyoruz.

Bit 3 **GPIODEN**: IO port D clock enable

Set and cleared by software.

0: IO port D clock disabled

1: IO port D clock enabled

RCC->AHB1ENR |= (1 << 3);

- Pinleri set ya da reset edeceğimiz kısımdır. Bunları while döngüsü içerisinde yazıyoruz.

### GPIO port output data register (GPIOx\_ODR) (x = A..I/J/K)

Address offset: 0x14

Reset value: 0x0000 0000

| 31       | 30    | 29    | 28    | 27    | 26    | 25   | 24   | 23   | 22   | 21   | 20   | 19   | 18   | 17   | 16   |
|----------|-------|-------|-------|-------|-------|------|------|------|------|------|------|------|------|------|------|
| Reserved |       |       |       |       |       |      |      |      |      |      |      |      |      |      |      |
| 15       | 14    | 13    | 12    | 11    | 10    | 9    | 8    | 7    | 6    | 5    | 4    | 3    | 2    | 1    | 0    |
| ODR15    | ODR14 | ODR13 | ODR12 | ODR11 | ODR10 | ODR9 | ODR8 | ODR7 | ODR6 | ODR5 | ODR4 | ODR3 | ODR2 | ODR1 | ODR0 |
| rw       | rw    | rw    | rw    | rw    | rw    | rw   | rw   | rw   | rw   | rw   | rw   | rw   | rw   | rw   | rw   |

Bits 31:16 Reserved, must be kept at reset value.

Bits 15:0 **ODRy**: Port output data (y = 0..15)

These bits can be read and written by software.

Note: For atomic bit set/reset, the ODR bits can be individually set and reset by writing to the GPIOx\_BSRR register (x = A..I/J/K).

```
GPIOD->ODR |= 1 << 12;
GPIOD->ODR |= 1 << 13;
GPIOD->ODR |= 1 << 14;
GPIOD->ODR |= 1 << 15;
```

- Clock ayarlarını aşağıdaki gibi düzelttik.
- Önceki yapıtlarımızın hepsini HEX formatında olacak şekilde düzelttik.
- Konunun videosu eski olduğundan önceki yaptığımız örneklerden farklılık olarak eskilerde RCC\_CFGR register kısmın 4:7 arası bitler koda eklenmişken yenilerde ekli olan 0:3 arası bitler eklenmemiş.

Eski videoda döngüde açılan flagların kapatılması için RCC\_CIR registerin 11. ve 23.bitleri resetlemiştir.

```
8 void RCC_Config(void)
9 {
10    RCC->CR |= 0x00030000;           //HSEON, HSERDY
11    while(!(RCC->CR & 0x00020000)); //HSERDY
12    RCC->CR |= 0x00080000;          //CSSON
13    RCC->CFGR = 0x00000000;
14    RCC->PLLCFGR |= 0x00400000;    //PLLSRC
15    RCC->PLLCFGR |= 0x00000004;    //PLLM 4
16    RCC->PLLCFGR |= 0x00002A00;    //PLLN 168
17    RCC->PLLCFGR |= 0x00000000;    //PLLP 2
18    RCC->CR |= 0x01000000;          //PLLON
19    while(!(RCC->CR & 0x02000000)); //PLLRDY
20    RCC->CFGR |= 0x00000001;        //SW
21    while(!(RCC->CR & 0x00000001)); //SWS
22 }
```

- GPIO ayarlarını aşağıdaki gibi düzelttik.

```

19 void GPIO_Config(void)
20 {
21     RCC->AHB1ENR |= (1 << 3);           //D clock enable
22
23     //output mode
24     GPIOD->MODER |= 0x55000000;        //PD12, PD13, PD14, PD15
25
26     //Very high speed
27     GPIOD->OSPEEDR |= 0xFF000000;
28 }

```

### Kod Kısmı

- RCC ve GPIO için yazdığımız fonksiyonları ekledik ardından while dönüşü içerisinde led blink uygulamasını gerçekleştirdik.

```

36 int main(void)
37 {
38
39     RCC_Config();
40     SystemCoreClockUpdate();
41
42     GPIO_Config();
43
44     while (1)
45     {
46         //set
47         GPIOD->ODR |= 1 << 12;
48         GPIOD->ODR |= 1 << 13;
49         GPIOD->ODR |= 1 << 14;
50         GPIOD->ODR |= 1 << 15;
51
52         for(int i=0; i<1680000; i++);
53
54         //reset
55         GPIOD->ODR &= ~(1 << 12);
56         GPIOD->ODR &= ~(1 << 13);
57         GPIOD->ODR &= ~(1 << 14);
58         GPIOD->ODR &= ~(1 << 15);
59
60         for(int i=0; i<1680000; i++);
61     }
62 }

```

01 02 Buton ile Led Yakma

25 Aralık 2021 Cumartesi 00:52

## 01\_02 Buton ile Led Yakma

➤ HAL

## Konfigürasyon Kısımları

- Kullanıcı butonu C portun 13.pinine bağlı.



| Pin Name | Signal on Pin | GPIO output | GPIO mode      | GPIO Pull-up    | Maximum o. | User Label | Modified |
|----------|---------------|-------------|----------------|-----------------|------------|------------|----------|
| PB3      | n/a           | Low         | Output Push... | No pull-up a... | Very High  | LED1       | ✓        |
| PB4      | n/a           | Low         | Output Push... | No pull-up a... | Very High  | LED2       | ✓        |
| PB5      | n/a           | Low         | Output Push... | No pull-up a... | Very High  | LED3       | ✓        |
| PB6      | n/a           | Low         | Output Push... | No pull-up a... | Very High  | LED4       | ✓        |
| PC13     | n/a           | n/a         | Input mode     | Pull-down       | n/a        | BUTTON     | ✓        |

Kod Kısmı

- RCC için bir değişiklik yapmadığımızdan fonksiyon içeriği aynıdır.
  - hal\_gpio.c kısmından ReadPin fonksiyonlarına ulaşabiliriz.

```
    HAL_GPIO_RisingPinReadInFunctionCallback(IRQn);
}
}

375 GPIO_PinState HAL_GPIO_ReadPin(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin)
376 {
377     GPIO_PinState bitstatus;
378
379     /* Check the parameters */
380     assert_param(IS_GPIO_PIN(GPIO_Pin));
381
382     if((GPIOx->IDR & GPIO_Pin) != (uint32_t)GPIO_PIN_RESET)
383     {
384         bitstatus = GPIO_PIN_SET;
385     }
386     else
387     {
388         bitstatus = GPIO_PIN_RESET;
389     }
390
391     return bitstatus;
392 }
```

- Buton için count değişkeni atıyoruz.

```
44 /* USER CODE BEGIN PV */  
45 int count=0;  
46 /* USER CODE END PV */
```

- Okuma yaptığında yani butona bastığımızda if yapısının içine girer ve while ile kullanıcının eli butona basılı olup olmadığını kontrol ederiz. Elini çektiğinde count değerini 1 arttırıyor.
  - Count değerinin modunu aldığımda 0 iken RESET, 1 iken SET durumundadır.

```
• Count degerin modunu aldiginizda  
94 while (1)  
95 {  
96     /* USER CODE END WHILE */  
97  
98     /* USER CODE BEGIN 3 */
```

```

94     while (1)
95     {
96         /* USER CODE END WHILE */
97
98         /* USER CODE BEGIN 3 */
99         if(HAL_GPIO_ReadPin(GPIOC, BUTTON_Pin))
100        {
101            while(HAL_GPIO_ReadPin(GPIOC, BUTTON_Pin));
102            HAL_Delay(100);
103            count++;
104        }
105
106        if(count % 2 == 1)
107        {
108            HAL_GPIO_WritePin(GPIOB, LED1_Pin | LED2_Pin | LED3_Pin | LED4_Pin,GPIO_PIN_SET);
109        }
110        else
111        {
112            HAL_GPIO_WritePin(GPIOB, LED1_Pin | LED2_Pin | LED3_Pin | LED4_Pin,GPIO_PIN_RESET);
113        }
114    }

```

## ➤ REGISTER

### Konfigürasyon Kısımlı

- Öncelikle RCC ve GPIO ayarlarını yapıyoruz.

```

8 void RCC_Config(void)
9 {
10    RCC->CR |= 0x00030000;           //HSEON, HSERDY
11    while(!(RCC->CR & 0x00020000)); //HSERDY
12    RCC->CR |= 0x00080000;          //CSSON
13    RCC->CFGR = 0x00000000;
14    RCC->PLLCFGR |= 0x00400000;    //PLLSRC
15    RCC->PLLCFGR |= 0x00000004;    //PLLM 4
16    RCC->PLLCFGR |= 0x00002A00;    //PLLN 168
17    RCC->PLLCFGR |= 0x00000000;    //PLLP 2
18    RCC->CR |= 0x01000000;          //PLLON
19    while(!(RCC->CR & 0x02000000)); //PLLRDY
20    RCC->CFGR |= 0x00000001;        //SW
21    while(!(RCC->CR & 0x00000001)); //SWS
22 }

```

- GPIO ayarlarını aşağıdaki gibi düzelttik.
- A portunu da kullanacağımızdan A ve D portlarını RCC clocklarını aktif ediyoruz.

```
RCC->AHB1ENR |= 0x00000009;           //A, D clock enable
```

```

24 void GPIO_Config(void)
25 {
26    RCC->AHB1ENR |= 0x00000009;           //A, D clock enable
27
28    GPIOD->MODER |= 0x55000000;          //PD12, PD13, PD14, PD15
29    GPIOD->OTYPER |= 0x00000000;         //Output push-pull
30    GPIOD->OSPEEDR |= 0xFF000000;        //Very high speed
31    GPIOD->PUPDR |= 0x00000000;          //No pull-up, pull-down
32 }

```

### Kod Kısımlı

- Buton için A0 pinı için okuma yapacağız. Burada ilk biti kullanacağız. While döngüsü içinde bir if yapısı içinde eğer okuma yapılrsa while döngüsüne girer ve okuma yani butona basma devam ediyorsa count değişkenini 1 arttırır.
- Buradaki delay ark olayından dolayı kullandık.

### GPIO port input data register (GPIOx\_IDR) (x = A..I/J/K)

Address offset: 0x10

Reset value: 0x0000 XXXX (where X means undefined)

| 31       | 30    | 29    | 28    | 27    | 26    | 25   | 24   | 23   | 22   | 21   | 20   | 19   | 18   | 17   | 16   |
|----------|-------|-------|-------|-------|-------|------|------|------|------|------|------|------|------|------|------|
| Reserved |       |       |       |       |       |      |      |      |      |      |      |      |      |      |      |
| 15       | 14    | 13    | 12    | 11    | 10    | 9    | 8    | 7    | 6    | 5    | 4    | 3    | 2    | 1    | 0    |
| IDR15    | IDR14 | IDR13 | IDR12 | IDR11 | IDR10 | IDR9 | IDR8 | IDR7 | IDR6 | IDR5 | IDR4 | IDR3 | IDR2 | IDR1 | IDR0 |

Bits 31:16 Reserved, must be kept at reset value.

Bits 15:0 IDRy: Port input data (y = 0..15)

These bits are read-only and can be accessed in word mode only. They contain the input value of the corresponding I/O port.

```
if(GPIOA->IDR & 0x00000001)
{
    while(GPIOA->IDR & 0x00000001);
    delay(1680000);

    count++;
}
```

- Count ve delay için globalde tanıttık.
- While döngüsünün içi 1 olduğu sürece çalışmaya devam eder ancak 0 olduğunda döngüden çıkış alt satırı geçer.

```
3 int count = 0;
4
5 void delay(uint32_t time)
6 {
7     while(time--);
8 }
```

- Count değişkenini kalanı 0 ise yani 2.defa basıldığında ledi söndürken kalan 1 olduğunda yani 1.defa basıldığı durumda ledi yakacaktır.

```
57 int main(void)
58 {
59     RCC_Config();
60     SystemCoreClockUpdate();
61
62     GPIO_Config();
63
64     while (1)
65     {
66         if(GPIOA->IDR & 0x00000001)
67         {
68             while(GPIOA->IDR & 0x00000001);
69             delay(1680000);
70
71             count++;
72         }
73         if(count %2 == 0)
74             GPIOD->ODR |= 0x00000000;
75         else
76             GPIOD->ODR |= 0x0000F000;
77     }
78 }
```

## 02\_01 External Interrupt

25 Aralık 2021 Cumartesi 00:54

## 02\_01 External Interrupt



## Konfigürasyon Kısmı



| Pin Name | Signal on Pin | GPIO output | GPIO mode        | GPIO Pull-up    | Maximum ou... | User Label | Modified |
|----------|---------------|-------------|------------------|-----------------|---------------|------------|----------|
| PB3      | n/a           | Low         | Output Push...   | No pull-up a... | Very High     | LED1       | ✓        |
| PB4      | n/a           | Low         | Output Push...   | No pull-up a... | Very High     | LED2       | ✓        |
| PB5      | n/a           | Low         | Output Push...   | No pull-up a... | Very High     | LED3       | ✓        |
| PB6      | n/a           | Low         | Output Push...   | No pull-up a... | Very High     | LED4       | ✓        |
| PC0      | n/a           | n/a         | External Inte... | Pull-down       | n/a           | BUTTON1    | ✓        |
| PC1      | n/a           | n/a         | External Inte... | Pull-down       | n/a           | BUTTON2    | ✓        |
| PC2      | n/a           | n/a         | External Inte... | Pull-down       | n/a           | BUTTON3    | ✓        |

- NVIC kısmından işaretlediğimiz pinlerin kesme işlemi yapabilmesi için Enabled kısmı işaretliyoruz.

| NVIC Interrupt Table  | Enabled                             | Preemption Priority | Sub Priority |
|-----------------------|-------------------------------------|---------------------|--------------|
| EXTI line 0 interrupt | <input checked="" type="checkbox"/> | 0                   | 0            |
| EXTI line 1 interrupt | <input checked="" type="checkbox"/> | 0                   | 0            |
| EXTI line 2 interrupt | <input checked="" type="checkbox"/> | 0                   | 0            |

Kod Kismi

- İt.c dosyasına gittiğimizde 3 adet kesme fonksiyonları oluşturulmuştur.

```
205 void EXTI0_IRQHandler(void)
206 {
207     /* USER CODE BEGIN EXTI0_IRQn_0 */
208
209     /* USER CODE END EXTI0_IRQn_0 */
210     HAL_GPIO_EXTI_IRQHandler(GPIO_PIN_0);
211     /* USER CODE BEGIN EXTI0_IRQn_1 */
212
213     /* USER CODE END EXTI0_IRQn_1 */
```

```

205 void EXTI0_IRQHandler(void)
206 {
207     /* USER CODE BEGIN EXTI0_IRQn 0 */
208
209     /* USER CODE END EXTI0_IRQn 0 */
210     HAL_GPIO_EXTI_IRQHandler(GPIO_PIN_0);
211     /* USER CODE BEGIN EXTI0_IRQn 1 */
212
213     /* USER CODE END EXTI0_IRQn 1 */
214 }

219 void EXTI1_IRQHandler(void)
220 {
221     /* USER CODE BEGIN EXTI1_IRQn 0 */
222
223     /* USER CODE END EXTI1_IRQn 0 */
224     HAL_GPIO_EXTI_IRQHandler(GPIO_PIN_1);
225     /* USER CODE BEGIN EXTI1_IRQn 1 */
226
227     /* USER CODE END EXTI1_IRQn 1 */
228 }

233 void EXTI2_IRQHandler(void)
234 {
235     /* USER CODE BEGIN EXTI2_IRQn 0 */
236
237     /* USER CODE END EXTI2_IRQn 0 */
238     HAL_GPIO_EXTI_IRQHandler(GPIO_PIN_2);
239     /* USER CODE BEGIN EXTI2_IRQn 1 */
240
241     /* USER CODE END EXTI2_IRQn 1 */
242 }

```

- İçerisinde yazılı olana Ctrl + Space tıkladığımızda hal\_gpio.c dosyaya götürür. Burada en son satırda Callback fonksiyonunu çağırır.

```

492 void HAL_GPIO_EXTI_IRQHandler(uint16_t GPIO_Pin)
493 {
494     /* EXTI line interrupt detected */
495     if(__HAL_GPIO_EXTI_GET_IT(GPIO_Pin) != RESET)
496     {
497         __HAL_GPIO_EXTI_CLEAR_IT(GPIO_Pin);
498         HAL_GPIO_EXTI_Callback(GPIO_Pin);
499     }

```

- Buna aynı şekilde gittiğimizde bu dosya üzerinde yer alır. Biz main.c dosyamızda bu fonksiyonu kullanıp kesme işlemi yapacağız.

```

507 __weak void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
508 {
509     /* Prevent unused argument(s) compilation warning */
510     UNUSED(GPIO_Pin);
511     /* NOTE: This function Should not be modified, when the callback is needed,
512             the HAL_GPIO_EXTI_Callback could be implemented in the user file
513     */
514 }

```

- While döngüsü ile tüm pinleri yaktık ve count değişkeni değerinde delay süresi verdik.

```

23 /* Private includes -----
24 /* USER CODE BEGIN Includes */
25 int count=0 , i=0;
26 /* USER CODE END Includes */

```

```

125     while (1)
126     {
127         /* USER CODE END WHILE */
128
129         /* USER CODE BEGIN 3 */
130         HAL_GPIO_TogglePin(GPIOB, GPIO_PIN_All);
131         HAL_Delay(count);
132     }
133     /* USER CODE END 3 */
134 }
```

- Count değişkenini kesme olduğunda her buton için ayrı delay süreleri verdik.

```

55/* Private user code -----
56 /* USER CODE BEGIN 0 */
57void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
58 {
59     i++;
60     if(HAL_GPIO_ReadPin(GPIOC, BUTTON1_Pin))
61     {
62         if(i==1)
63         {
64             count=1000;
65         }
66         else if(i==2)
67         {
68             count=750;
69         }
70         else if(i==3)
71         {
72             count=500;
73         }
74     else
75     {
76         count=250;
77         i=0;
78     }
79 }
80 else if(HAL_GPIO_ReadPin(GPIOC, BUTTON2_Pin))
81 {
82     count=100;
83 }
84 else
85 {
86     count=50;
87 }
88 }
89 /* USER CODE END 0 */
```

## ➤ REGISTER

### Konfigürasyon Kısmı

- External Interrup'tın clock hattı APB2'ye gidiyor.



## RCC APB2 peripheral clock enable register (RCC\_APB2ENR)

Address offset: 0x44

Reset value: 0x0000 0000

Access: no wait state, word, half-word and byte access.

|               |                |               |            |            |            |            |            |          |    |               |               |          |    |            |            |         |
|---------------|----------------|---------------|------------|------------|------------|------------|------------|----------|----|---------------|---------------|----------|----|------------|------------|---------|
| 31            | 30             | 29            | 28         | 27         | 26         | 25         | 24         | 23       | 22 | 21            | 20            | 19       | 18 | 17         | 16         |         |
| Reserved      |                |               |            |            |            |            |            |          |    |               |               |          |    | TIM11 EN   | TIM10 EN   | TIM9 EN |
|               |                |               |            |            |            |            |            |          |    | rw            | rw            | rw       |    |            |            |         |
| 15            | 14             | 13            | 12         | 11         | 10         | 9          | 8          | 7        | 6  | 5             | 4             | 3        | 2  | 1          | 0          |         |
| Reser-<br>ved | SYSCF<br>G GEN | Reser-<br>ved | SPI1<br>EN | SDIO<br>EN | ADC3<br>EN | ADC2<br>EN | ADC1<br>EN | Reserved |    | USART<br>6 EN | USART<br>1 EN | Reserved |    | TIM8<br>EN | TIM1<br>EN |         |
|               | rw             |               | rw         | rw         | rw         | rw         | rw         |          |    | rw            | rw            |          |    | rw         | rw         |         |

Bit 14 **SYSCFGGEN**: System configuration controller clock enable

Set and cleared by software.

0: System configuration controller clock disabled

1: System configuration controller clock enabled

- 14.pini 1 yapıyoruz.

RCC->**APB2ENR** |= 0x00004000; //SYSCGFEN

- 3 tane interrupt pini kullanacağımızdan öncelikle bunları aktif edip, öncelik sırası belirlemeliyiz.
- core\_cm4.h kütüphanesinde bu fonksiyonları görebiliriz.
- İkinci değere öncelik numarası yazıyoruz. Ne kadar küçük olursa öncelik

|1452@ \_\_STATIC\_INLINE void NVIC\_EnableIRQ(IRQn\_Type IRQn)

sıralamasında önde olur.

```
1535@ __STATIC_INLINE void NVIC_SetPriority(IRQn_Type IRQn, uint32_t priority)
```

```
NVIC_EnableIRQ(EXTI0_IRQn);  
NVIC_EnableIRQ(EXTI1_IRQn);  
NVIC_EnableIRQ(EXTI2_IRQn);  
  
NVIC_SetPriority(EXTI0_IRQn, 0);  
NVIC_SetPriority(EXTI0_IRQn, 1);  
NVIC_SetPriority(EXTI0_IRQn, 2);
```

- Interrupt olarak mı event olarak mı kullanacağımızı belirliyoruz. Interrupt olarak kullanacağız.

### Interrupt mask register (EXTI\_IMR)

Address offset: 0x00

Reset value: 0x0000 0000

| 31       | 30   | 29   | 28   | 27   | 26   | 25  | 24  | 23  | 22  | 21  | 20  | 19  | 18  | 17   | 16   |
|----------|------|------|------|------|------|-----|-----|-----|-----|-----|-----|-----|-----|------|------|
| Reserved |      |      |      |      |      |     |     |     |     |     |     |     |     | MR22 | MR21 |
|          |      |      |      |      |      |     |     |     | rw  | rw  | rw  | rw  | rw  | rw   | rw   |
| 15       | 14   | 13   | 12   | 11   | 10   | 9   | 8   | 7   | 6   | 5   | 4   | 3   | 2   | 1    | 0    |
| MR15     | MR14 | MR13 | MR12 | MR11 | MR10 | MR9 | MR8 | MR7 | MR6 | MR5 | MR4 | MR3 | MR2 | MR1  | MR0  |
| rw       | rw   | rw   | rw   | rw   | rw   | rw  | rw  | rw  | rw  | rw  | rw  | rw  | rw  | rw   | rw   |

Bits 31:23 Reserved, must be kept at reset value.

Bits 22:0 **MRx**: Interrupt mask on line x

- 0: Interrupt request from line x is masked  
1: Interrupt request from line x is not masked

- İlk 3 pine 1 yazıyoruz.

```
EXTI->EMR |= x00000007;
```

- Alçalan kenar mı yükselen kenar mı kullanacağımızı belirliyoruz. Yükselen kenar kullanacağız.

### Rising trigger selection register (EXTI\_RTSR)

Address offset: 0x08

Reset value: 0x0000 0000

| 31       | 30   | 29   | 28   | 27   | 26   | 25  | 24  | 23  | 22  | 21  | 20  | 19  | 18  | 17   | 16   |
|----------|------|------|------|------|------|-----|-----|-----|-----|-----|-----|-----|-----|------|------|
| Reserved |      |      |      |      |      |     |     |     |     |     |     |     |     | TR22 | TR21 |
|          |      |      |      |      |      |     |     |     | rw  | rw  | rw  | rw  | rw  | rw   | rw   |
| 15       | 14   | 13   | 12   | 11   | 10   | 9   | 8   | 7   | 6   | 5   | 4   | 3   | 2   | 1    | 0    |
| TR15     | TR14 | TR13 | TR12 | TR11 | TR10 | TR9 | TR8 | TR7 | TR6 | TR5 | TR4 | TR3 | TR2 | TR1  | TR0  |
| rw       | rw   | rw   | rw   | rw   | rw   | rw  | rw  | rw  | rw  | rw  | rw  | rw  | rw  | rw   | rw   |

Bits 31:23 Reserved, must be kept at reset value.

Bits 22:0 **TRx**: Rising trigger event configuration bit of line x

- 0: Rising trigger disabled (for Event and Interrupt) for input line  
1: Rising trigger enabled (for Event and Interrupt) for input line

İlk 3 pine 1 yazıyoruz.

```
EXTI->RTSR |= x00000007;
```

- Butonları external interrupt için aktif etmemiz gerekiyor.

Buton için A portunu kullanıyoruz.

## SYSCFG external interrupt configuration register 1 (SYSCFG\_EXTICR1)

Address offset: 0x08

Reset value: 0x0000 0000

|            |    |    |    |            |    |    |    |            |    |    |    |            |    |    |    |
|------------|----|----|----|------------|----|----|----|------------|----|----|----|------------|----|----|----|
| 31         | 30 | 29 | 28 | 27         | 26 | 25 | 24 | 23         | 22 | 21 | 20 | 19         | 18 | 17 | 16 |
| Reserved   |    |    |    |            |    |    |    |            |    |    |    |            |    |    |    |
| 15         | 14 | 13 | 12 | 11         | 10 | 9  | 8  | 7          | 6  | 5  | 4  | 3          | 2  | 1  | 0  |
| EXTI3[3:0] |    |    |    | EXTI2[3:0] |    |    |    | EXTI1[3:0] |    |    |    | EXTI0[3:0] |    |    |    |
| rw         | rw | rw | rw | rw         | rw | rw | rw | rw         | rw | rw | rw | rw         | rw | rw | rw |

Bits 31:16 Reserved, must be kept at reset value.

Bits 15:0 **EXTIx[3:0]**: EXTI x configuration (x = 0 to 3)

These bits are written by software to select the source input for the EXTIx external interrupt.

- 0000: PA[x] pin
- 0001: PB[x] pin
- 0010: PC[x] pin
- 0011: PD[x] pin
- 0100: PE[x] pin
- 0101: PF[x] pin
- 0110: PG[x] pin
- 0111: PH[x] pin
- 1000: PI[x] pin

SYSCFG->EXTICR[0] = 0x00000000;

- RCC, GPIO ve EXTI için yazdığımız fonksiyonlar aşağıdaki gibidir.

```
8 void RCC_Config(void)
9 {
10    RCC->CR |= 0x00030000;           //HSEON, HSERDY
11    while(!(RCC->CR & 0x00020000)); //HSERDY
12    RCC->CR |= 0x00080000;          //CSSON
13    RCC->CFGR = 0x00000000;
14    RCC->PLLCFGR |= 0x00040000;     //PLLSRC
15    RCC->PLLCFGR |= 0x00000004;     //PLLM 4
16    RCC->PLLCFGR |= 0x00002A00;     //PLLN 168
17    RCC->PLLCFGR |= 0x00000000;     //PLLP 2
18    RCC->CR |= 0x01000000;          //PLLON
19    while(!(RCC->CR & 0x02000000)); //PLLRDY
20    RCC->CFGR |= 0x00000001;        //SW
21    while(!(RCC->CR & 0x00000001)); //SWS
22 }

24 void GPIO_Config(void)
25 {
26    RCC->AHB1ENR |= 0x00000009;      //A, D clock enable
27
28    GPIOD->MODER |= 0x55000000;     //PD12, PD13, PD14, PD15
29    GPIOD->OTYPER |= 0x00000000;    //Output push-pull
30    GPIOD->OSPEEDR |= 0xFF000000;   //Very high speed
31    GPIOD->PUPDR |= 0x00000000;     //No pull-up, pull-down
32 }
```

```

34 void EXTI_Config(void)
35 {
36     RCC->APB2ENR |= 0x00004000;           //SYSCGFEN
37
38     NVIC_EnableIRQ(EXTI0_IRQn);
39     NVIC_EnableIRQ(EXTI1_IRQn);
40     NVIC_EnableIRQ(EXTI2_IRQn);
41
42     NVIC_SetPriority(EXTI0_IRQn, 0);
43     NVIC_SetPriority(EXTI0_IRQn, 1);
44     NVIC_SetPriority(EXTI0_IRQn, 2);
45
46     EXTI->EMR |= 0x00000007;
47     EXTI->RTSR |= 0x00000007;
48
49     SYSCFG->EXTICR[0] = 0x00000000;
50 }

```

## Kod Kısmı

- İlk önce while döngüsünde tüm ledleri açıyoruz.

```

82 int main(void)
83 {
84     RCC_Config();
85     SystemCoreClockUpdate();
86     GPIO_Config();
87     EXTI_Config();
88
89     while (1)
90     {
91         GPIOD->ODR |= 0x0000F000;
92     }
93 }

```

- Pinde kesme olup olmadığını öğrenmemiz gerekiyor. Bitte değer olduğunda kesmeye girmiş oluyor. Tekrar 1 yazarak bayrağı indiriyoruz.

## Pending register (EXTI\_PR)

Address offset: 0x14

Reset value: undefined

| 31       | 30    | 29    | 28    | 27    | 26    | 25    | 24    | 23    | 22    | 21    | 20    | 19    | 18    | 17    | 16    |      |
|----------|-------|-------|-------|-------|-------|-------|-------|-------|-------|-------|-------|-------|-------|-------|-------|------|
| Reserved |       |       |       |       |       |       |       |       |       | PR22  | PR21  | PR20  | PR19  | PR18  | PR17  | PR16 |
| 15       | 14    | 13    | 12    | 11    | 10    | 9     | 8     | 7     | 6     | 5     | 4     | 3     | 2     | 1     | 0     |      |
| PR15     | PR14  | PR13  | PR12  | PR11  | PR10  | PR9   | PR8   | PR7   | PR6   | PR5   | PR4   | PR3   | PR2   | PR1   | PR0   |      |
| rc_w1    | rc_w1 | rc_w1 | rc_w1 | rc_w1 | rc_w1 | rc_w1 | rc_w1 | rc_w1 | rc_w1 | rc_w1 | rc_w1 | rc_w1 | rc_w1 | rc_w1 | rc_w1 |      |

Bits 31:23 Reserved, must be kept at reset value.

Bits 22:0 PRx: Pending bit

0: No trigger request occurred

1: selected trigger request occurred

This bit is set when the selected edge event arrives on the external interrupt line.

This bit is cleared by programming it to '1'.

- Eğer kesmeye giriliyorsa 2 saniye boyunca belirlediğimiz ledi açıp diğerlerini kapatıyoruz.

```
51 void EXTI0_IRQHandler()
52 {
53     if(EXTI->PR & 0x00000001)
54     {
55         GPIOD->ODR |= 0x00001000;
56         delay(33600000);
57     }
58     EXTI->PR = 0x00000001;
59 }

61 void EXTI1_IRQHandler()
62 {
63     if(EXTI->PR & 0x00000002)
64     {
65         GPIOD->ODR |= 0x00002000;
66         delay(33600000);
67     }
68     EXTI->PR = 0x00000002;
69 }

71 void EXTI2_IRQHandler()
72 {
73     if(EXTI->PR & 0x00000004)
74     {
75         GPIOD->ODR |= 0x00004000;
76         delay(33600000);
77     }
78     EXTI->PR = 0x00000004;
79 }
```

03\_01 ADC Verisi Okuma

25 Aralık 2021 Cumartesi 00:56

## 03 01 ADC Verisi Okuma

➤ HAL

## Konfigürasyon Kısımları



| Pin Name | Signal on Pin | GPIO output | GPIO mode   | GPIO Pull-up/D              | Maximum Value | User Label | Modified                 |
|----------|---------------|-------------|-------------|-----------------------------|---------------|------------|--------------------------|
| PA0-WKUP | ADC1_IN0      | n/a         | Analog mode | No pull-up and no pull-down | n/a           |            | <input type="checkbox"/> |

- Analog değer okumak için potansiyometreyi PA0'a bağladık. Bunun için Analog kısmından ADC1'den IN0 tıklarız. Gerçek voltaj değerini de hesaplayacağımızdan Vrefinit Channel'da seçilir.

- IN0
  - IN1
  - IN2
  - IN3
  - IN4
  - IN5
  - IN6
  - IN7
  - IN8
  - IN9
  - IN10
  - IN11
  - IN12
  - IN13
  - IN14
  - IN15
  - Temperature Sensor Channel

- Daha sonra Parameter Settings'den Continuous Conversion Mode Enabled yapılır. Sürekli çevrim modu anlamına gelir ve tekrar tekrar okuma durumu yapar eğer çalışmazsa bir kere okur daha sonra değer okumaz.
  - Çözünürlüğümüzü 12 bit yaptık yani en fazla 4095 değerini görebiliriz
  - Çok kanal olduğundan Scan Mode Enabled yapılır.

### ADC Settings

|                               |                               |
|-------------------------------|-------------------------------|
| <u>Settings</u>               |                               |
| Clock Prescaler               | PCLK2 divided by 4            |
| Resolution                    | 12 bits (15 ADC Clock cycles) |
| Data Alignment                | Right alignment               |
| Scan Conversion Mode          | Enabled                       |
| Continuous Conversion Mode    | Enabled                       |
| Discontinuous Conversion Mode | Disabled                      |

|                               |                                                |
|-------------------------------|------------------------------------------------|
| ADC_Settings                  |                                                |
| Clock Prescaler               | PCLK2 divided by 4                             |
| Resolution                    | 12 bits (15 ADC Clock cycles)                  |
| Data Alignment                | Right alignment                                |
| Scan Conversion Mode          | Enabled                                        |
| Continuous Conversion Mode    | Enabled                                        |
| Discontinuous Conversion Mode | Disabled                                       |
| DMA Continuous Requests       | Disabled                                       |
| End Of Conversion Selection   | EOC flag at the end of single channel converts |

- Number Of Conversion kısmı 2 kanal olduğundan iki yazılır ardından rank kısmından öncelik sırası verilir.
- Sampling Time ile kaç cycle'da bir çevrim yapacağımızı belirtiyoruz.

|                                    |                                         |
|------------------------------------|-----------------------------------------|
| ADC-Regular_ConversionMode         |                                         |
| Number Of Conversion               | 2                                       |
| External Trigger Conversion Source | Regular Conversion launched by software |
| External Trigger Conversion Edge   | None                                    |
| Rank                               | 1                                       |
| Channel                            | Channel 0                               |
| Sampling Time                      | 56 Cycles                               |
| Rank                               | 2                                       |
| Channel                            | Channel Vrefint                         |
| Sampling Time                      | 56 Cycles                               |

### Kod Kısmı

- ADC okumada 3 mod var. Okumada Interrupt ve DMA kullanmayacağız. Polling Mode kullanımını yapacağız. Bunun nasıl kullanıldığını öğrenmek için hal\_adc.c dosyasından bakıyoruz.

```

87    *** Polling mode IO operation ***
88    =====
89    [..]
90    (+) Start the ADC peripheral using HAL_ADC_Start()
91    (+) Wait for end of conversion using HAL_ADC_PollForConversion(), at this stage
92    user can specify the value of timeout according to his end application
93    (+) To read the ADC converted values, use the HAL_ADC_GetValue() function.
94    (+) Stop the ADC peripheral using HAL_ADC_Stop()

```

- ADC çevre birimini başlatmak için 717.satırındaki fonksiyon kullanılır.
- "\*" var ise adresleme için "&" işaretini koymamız gerekiyor.

717@HAL\_StatusTypeDef HAL\_ADC\_Start(ADC\_HandleTypeDef\* hadc)

- Çevrimin takibini 883.satırındaki fonksiyon ile yapıyoruz.
- Timeout için milisecond cinsinden istiyor.

883@HAL\_StatusTypeDef HAL\_ADC\_PollForConversion(ADC\_HandleTypeDef\* hadc, uint32\_t Timeout)

1566@uint32\_t HAL\_ADC\_GetValue(ADC\_HandleTypeDef\* hadc)

840@HAL\_StatusTypeDef HAL\_ADC\_Stop(ADC\_HandleTypeDef\* hadc)

- İşlemcinin beslemesi gerçekte 3.3V değildir. Bunun için ayrı hesap yapılması gereklidir. Hesap için 2 değer için birini ADC okumasından diğerini ise adres kısmından bulacağız.
- Adres kısmı için datasheet kısmından bakılması gereklidir.

Internal reference voltage calibration values

| Symbol                 | Parameter                                              | Memory address            |
|------------------------|--------------------------------------------------------|---------------------------|
| V <sub>REFIN_CAL</sub> | Raw data acquired at temperature of 30 °C VDDA = 3.3 V | 0x1FFF 7A2A - 0x1FFF 7A2B |

- Bu adres için kodda tanımlama yapılması gerekiyor.
- Tanımlama yaparken bizim kullanacağımız bit 12, adres biti 32 bittir. Bizim burada 32 britten 16 bite dönüştürme yapmamız gerekiyor.

```

23@/* Private includes -----
24 /* USER CODE BEGIN Includes */
25 #define VREFIN_CAL ((uint16_t*) ((uint32_t) 0x1FFF7A2A))
26
27 uint16_t adc_value[2];
28 float Vadc=0,Vdda=0;
29 int count=0;
30 /* USER CODE END Includes */

```

- İki kanalı adc\_value değişkenine dizi halinde yazdık. 0.dizi rank'daki ayara göre ilk kanal oluyor.
- Diğer kanalı okuması için count değişkeni koyduk.

```

60/* Private user code -----
61 /* USER CODE BEGIN 0 */
62 void Read_ADC()
63 {
64     HAL_ADC_Start(&hadc1);
65
66     if(HAL_ADC_PollForConversion(&hadc1, 100000) == HAL_OK)
67     {
68         adc_value[count] = HAL_ADC_GetValue(&hadc1);
69         count++;
70
71         if(count==2)
72         {
73             count=0;
74         }
75
76         //Vadc=3.3*adc_value/4095;
77
78         Vdda=3.3>(*VREFIN_CAL)/adc_value[1];
79         Vadc=Vdda*adc_value[0]/4095;
80     }
81     HAL_ADC_Stop(&hadc1);
82 }
83 /* USER CODE END 0 */

```

| Variable Name | Address/Expression | Read Value |
|---------------|--------------------|------------|
| Vadc          | 0x20000028         | 1.7121019  |
| Vdda          | 0x2000002c         | 2.9433491  |
| count         | 0x20000030         | 0          |

## ➤ REGISTER

### Konfigürasyon Kısmı

- ADC'nin clock hattı APB2'ye gidiyor.



## RCC APB2 peripheral clock enable register (RCC\_APB2ENR)

Address offset: 0x44

Reset value: 0x0000 0000

Access: no wait state, word, half-word and byte access.

|               |               |               |            |            |            |            |            |               |                  |                  |               |            |            |             |             |            |
|---------------|---------------|---------------|------------|------------|------------|------------|------------|---------------|------------------|------------------|---------------|------------|------------|-------------|-------------|------------|
| 31            | 30            | 29            | 28         | 27         | 26         | 25         | 24         | 23            | 22               | 21               | 20            | 19         | 18         | 17          | 16          |            |
| Reserved      |               |               |            |            |            |            |            |               |                  |                  |               |            |            | TIM11<br>EN | TIM10<br>EN | TIM9<br>EN |
|               |               |               |            |            |            |            |            |               |                  |                  |               |            |            | rw          | rw          | rw         |
| 15            | 14            | 13            | 12         | 11         | 10         | 9          | 8          | 7             | 6                | 5                | 4             | 3          | 2          | 1           | 0           |            |
| Reser-<br>ved | SYSCE<br>FGEN | Reser-<br>ved | SPI1<br>EN | SDIO<br>EN | ADC3<br>EN | ADC2<br>EN | ADC1<br>EN | Reser-<br>ved | USART<br>6<br>EN | USART<br>1<br>EN | Reser-<br>ved | TIM8<br>EN | TIM1<br>EN |             |             |            |
|               | rw            |               | rw         | rw         | rw         | rw         | rw         |               | rw               | rw               |               | rw         | rw         |             |             |            |

Bit 14 **SYSCFGGEN**: System configuration controller clock enable

Set and cleared by software.

0: System configuration controller clock disabled

1: System configuration controller clock enabled

- 8.pini 1 yapıyoruz.

RCC->APB2ENR |= 0x00000100; //ADC1

## ADC control register 1 (ADC\_CR1)

Address offset: 0x04

Reset value: 0x0000 0000

|              |             |            |       |             |      |        |       |       |            |    |    |    |    |          |    |  |
|--------------|-------------|------------|-------|-------------|------|--------|-------|-------|------------|----|----|----|----|----------|----|--|
| 31           | 30          | 29         | 28    | 27          | 26   | 25     | 24    | 23    | 22         | 21 | 20 | 19 | 18 | 17       | 16 |  |
| Reserved     |             |            |       |             |      |        |       |       |            |    |    |    |    | Reserved |    |  |
|              |             |            |       |             |      |        |       |       |            |    |    |    |    | rw       |    |  |
| 15           | 14          | 13         | 12    | 11          | 10   | 9      | 8     | 7     | 6          | 5  | 4  | 3  | 2  | 1        | 0  |  |
| DISCNUM[2:0] | JDISCE<br>N | DISC<br>EN | JAUTO | AWDSEL<br>L | SCAN | JEOCIE | AWDIE | EOCIE | AWDCH[4:0] |    |    |    |    |          |    |  |
| rw           | rw          | rw         | rw    | rw          | rw   | rw     | rw    | rw    | rw         | rw | rw | rw | rw | rw       | rw |  |

Bits 25:24 **RES[1:0]**: Resolution

These bits are written by software to select the resolution of the conversion.

00: 12-bit (15 ADCCLK cycles)

01: 10-bit (13 ADCCLK cycles)

10: 8-bit (11 ADCCLK cycles)

11: 6-bit (9 ADCCLK cycles)

- 8 bit çözünürlük ayarlıyoruz.

ADC1->**CR1** |= 0x02000000; //RES 8-bit

## ADC control register 2 (ADC\_CR2)

Address offset: 0x08

Reset value: 0x0000 0000

|          |    |             |       |       |             |     |     |          |          |              |        |    |              |    |      |      |
|----------|----|-------------|-------|-------|-------------|-----|-----|----------|----------|--------------|--------|----|--------------|----|------|------|
| 31       | 30 | 29          | 28    | 27    | 26          | 25  | 24  | 23       | 22       | 21           | 20     | 19 | 18           | 17 | 16   |      |
| reserved |    | SWST<br>ART | EXTEN |       | EXTSEL[3:0] |     |     |          | reserved | JSWST<br>ART | JEXTEN |    | JEXTSEL[3:0] |    |      |      |
|          |    | rw          | rw    | rw    | rw          | rw  | rw  | rw       |          | rw           | rw     | rw | rw           | rw | rw   |      |
| 15       | 14 | 13          | 12    | 11    | 10          | 9   | 8   | 7        | 6        | 5            | 4      | 3  | 2            | 1  | 0    |      |
| reserved |    |             |       | ALIGN | EOCS        | DDS | DMA | Reserved |          |              |        |    |              |    | CONT | ADON |
|          |    |             |       | rw    | rw          | rw  | rw  |          |          |              |        |    |              |    | rw   | rw   |

Bit 0 **ADON**: A/D Converter ON / OFF

This bit is set and cleared by software.

Note: 0: Disable ADC conversion and go to power down mode

1: Enable ADC

- ADC aktif etmek için ilk bit 1 yaptı.

ADC1->**CR2** |= 0x00000001; //ADON

- ADC\_SPMPR register kısmında kaç cycle'da bir çevrim yapacağımızı belirtiyoruz.
- 0'dan 18'e kadar kanal var.



### ADC sample time register 1 (ADC\_SMPR1)

Address offset: 0x0C

Reset value: 0x0000 0000

| 31       | 30         | 29 | 28 | 27         | 26 | 25 | 24         | 23 | 22 | 21         | 20 | 19 | 18         | 17 | 16 |
|----------|------------|----|----|------------|----|----|------------|----|----|------------|----|----|------------|----|----|
| Reserved |            |    |    | SMP18[2:0] |    |    | SMP17[2:0] |    |    | SMP16[2:0] |    |    | SMP15[2:1] |    |    |
|          |            |    |    | rw         | rw | rw | rw         | rw | rw | rw         | rw | rw | rw         | rw | rw |
| 15       | 14         | 13 | 12 | 11         | 10 | 9  | 8          | 7  | 6  | 5          | 4  | 3  | 2          | 1  | 0  |
| SMP15_0  | SMP14[2:0] |    |    | SMP13[2:0] |    |    | SMP12[2:0] |    |    | SMP11[2:0] |    |    | SMP10[2:0] |    |    |
| rw       | rw         | rw | rw | rw         | rw | rw | rw         | rw | rw | rw         | rw | rw | rw         | rw | rw |

### ADC sample time register 2 (ADC\_SMPR2)

Address offset: 0x10

Reset value: 0x0000 0000

| 31       | 30        | 29 | 28 | 27        | 26 | 25 | 24        | 23 | 22 | 21        | 20 | 19 | 18        | 17 | 16        |
|----------|-----------|----|----|-----------|----|----|-----------|----|----|-----------|----|----|-----------|----|-----------|
| Reserved |           |    |    | SMP9[2:0] |    |    | SMP8[2:0] |    |    | SMP7[2:0] |    |    | SMP6[2:0] |    | SMP5[2:1] |
|          |           |    |    | rw        | rw | rw | rw        | rw | rw | rw        | rw | rw | rw        | rw | rw        |
| 15       | 14        | 13 | 12 | 11        | 10 | 9  | 8         | 7  | 6  | 5         | 4  | 3  | 2         | 1  | 0         |
| SMP_0    | SMP4[2:0] |    |    | SMP3[2:0] |    |    | SMP2[2:0] |    |    | SMP1[2:0] |    |    | SMP0[2:0] |    |           |
| rw       | rw        | rw | rw | rw        | rw | rw | rw        | rw | rw | rw        | rw | rw | rw        | rw | rw        |

- Biz 0.pinde çalıştığımız için 0.kanal kullanıyoruz.

Bits 31:30 Reserved, must be kept at reset value.

Bits 29:0 **SMPx[2:0]**: Channel x sampling time selection

These bits are written by software to select the sampling time individually for each channel.

During sample cycles, the channel selection bits must remain unchanged.

Note: 000: 3 cycles  
001: 15 cycles  
010: 28 cycles  
011: 56 cycles  
100: 84 cycles  
101: 112 cycles  
110: 144 cycles  
111: 480 cycles

- 56 cycles'da çalışacağız.

ADC1->**SMPR2** |= 0x00000003; //SMP0 56 cycles

### ADC common control register (ADC\_CCR)

Address offset: 0x04 (this offset address is relative to ADC1 base address + 0x300)

Reset value: 0x0000 0000

| 31       | 30  | 29   | 28         | 27      | 26 | 25    | 24       | 23       | 22 | 21 | 20         | 19     | 18 | 17 | 16 |
|----------|-----|------|------------|---------|----|-------|----------|----------|----|----|------------|--------|----|----|----|
| Reserved |     |      |            | TSVREFE |    | VBATE |          | Reserved |    |    |            | ADCPRE |    |    |    |
|          |     |      |            | rw      | rw |       |          |          |    |    |            | rw     | rw |    |    |
| 15       | 14  | 13   | 12         | 11      | 10 | 9     | 8        | 7        | 6  | 5  | 4          | 3      | 2  | 1  | 0  |
| DMA[1:0] | DDS | Res. | DELAY[3:0] |         |    |       | Reserved |          |    |    | MULTI[4:0] |        |    |    |    |
| rw       | rw  | rw   | rw         | rw      | rw | rw    | rw       | rw       | rw | rw | rw         | rw     |    |    |    |

Bits 17:16 **ADCPRE**: ADC prescaler

Set and cleared by software to select the frequency of the clock to the ADC. The clock is common for all the ADCs.

Note: 00: PCLK2 divided by 2  
01: PCLK2 divided by 4  
10: PCLK2 divided by 6  
11: PCLK2 divided by 8

- ADC birimin maksimum clock hızı 36 MHz olması gerekiğinden CubeMX programında baktığımızda ADC birimin bağlı olduğu APB2 hattı 84MHz olduğundan bu clock hızını 4'e böldüğümüzde sağlamış oluyor. Bu sebeple 16. ve 17. bitleri ona göre düzenliyoruz.

ADC->**CCR** |= 0x00010000;

- RCC, GPIO ve ADC için yazdığımız fonksiyonlar aşağıdaki gibidir.

```

21 void GPIO_Config(void)
22 {
23     RCC->AHB1ENR |= 0x00000001;           //A clock enable
24
25     GPIOA->MODER |= 0x00000003;          //PA0 Analog mode
26     GPIOA->OSPEEDR |= 0xFF000000;        //Very high speed
27 }

50 void RCC_Config(void)
6 {
7     RCC->CR |= 0x00030000;                //HSEON, HSERDY
8     while(!(RCC->CR & 0x00020000));      //HSERDY
9     RCC->CR |= 0x00080000;                //CSSON
10    RCC->CFGR = 0x00000000;
11    RCC->PLLCFGR |= 0x00400000;          //PLLSRC
12    RCC->PLLCFGR |= 0x00000004;          //PLLM 4
13    RCC->PLLCFGR |= 0x00002A00;          //PLLN 168
14    RCC->PLLCFGR |= 0x00000000;          //PLLP 2
15    RCC->CR |= 0x01000000;                //PLLON
16    while(!(RCC->CR & 0x02000000));      //PLLRDY
17    RCC->CFGR |= 0x00000001;              //SW
18    while(!(RCC->CR & 0x00000001));      //SWS
19 }

29 void ADC_Config(void)
30 {
31     RCC->APB2ENR |= 0x00000100;          //ADC1
32
33     ADC1->CR1 |= 0x02000000;            //RES 8-bit
34     ADC1->CR2 |= 0x00000001;            //ADON
35     ADC1->SMPR2 |= 0x00000003;          //SMP0 56 cycles
36     ADC->CCR |= 0x00010000;             //ADCPRE 4
37 }

```

#### Kod Kısıtları

- 8 bitlik değişken okuyacağımızdan ona göre değişken ataması yapıyoruz.
- `uint8_t adc_value;`
- 8 bitlik dönüşüm değeri olan bir ADC okuma fonksiyonu yazıyoruz ve bunu while döngüsünde `adc_value` değerini `Read_ADC()` fonksiyonunda tutmak için eşitliyoruz.
- Fonksiyonun içine değişken ataması yaptık.

```

40 uint8_t Read_ADC()
41 {
42     uint8_t value;
43
44 }

adc_value = Read_ADC();

```

## ADC control register 2 (ADC\_CR2)

Address offset: 0x08

Reset value: 0x0000 0000

| 31       | 30          | 29    | 28 | 27          | 26   | 25  | 24  | 23       | 22           | 21     | 20 | 19           | 18   | 17 | 16 |
|----------|-------------|-------|----|-------------|------|-----|-----|----------|--------------|--------|----|--------------|------|----|----|
| reserved | SWST<br>ART | EXTEN |    | EXTSEL[3:0] |      |     |     | reserved | JSWST<br>ART | JEXTEN |    | JEXTSEL[3:0] |      |    |    |
|          | rw          | rw    | rw | rw          | rw   | rw  | rw  |          | rw           | rw     | rw | rw           | rw   | rw | rw |
| 15       | 14          | 13    | 12 | 11          | 10   | 9   | 8   | 7        | 6            | 5      | 4  | 3            | 2    | 1  | 0  |
| reserved |             |       |    | ALIGN       | EOCS | DDS | DMA | Reserved |              |        |    | CONT         | ADON |    |    |
|          |             |       |    | rw          | rw   | rw  | rw  |          |              |        |    | rw           | rw   |    |    |

- ADC'nin yazılımsal olarak çevrimi başlatmak için 30.biti 1 yapıyoruz.

Bit 30 SWSTART: Start conversion of regular channels

This bit is set by software to start conversion and cleared by hardware as soon as the conversion starts.

0: Reset state

1: Starts conversion of regular channels

Note: This bit can be set only when ADON = 1 otherwise no conversion is launched.

```
ADC1->CR2 |= 0x40000000;           //SWSTART
```

- Çevrimin bitip bitmediğini ADC\_SR registerinden öğreniyoruz.

## ADC status register (ADC\_SR)

Address offset: 0x00

Reset value: 0x0000 0000

|          |      |       |      |     |     |       |       |       |       |       |       |       |       |       |       |
|----------|------|-------|------|-----|-----|-------|-------|-------|-------|-------|-------|-------|-------|-------|-------|
| 31       | 30   | 29    | 28   | 27  | 26  | 25    | 24    | 23    | 22    | 21    | 20    | 19    | 18    | 17    | 16    |
| Reserved |      |       |      |     |     |       |       |       |       |       |       |       |       |       |       |
| 15       | 14   | 13    | 12   | 11  | 10  | 9     | 8     | 7     | 6     | 5     | 4     | 3     | 2     | 1     | 0     |
| Reserved |      |       |      |     |     |       |       |       |       |       |       |       |       |       |       |
| OVR      | STRT | JSTRT | JEOC | EOC | AWD | rc_w0 | rc_w0 | rc_w0 | rc_w0 | rc_w0 | rc_w0 | rc_w0 | rc_w0 | rc_w0 | rc_w0 |

- 1.bit 1 oldu ise çevrim bittiği anlamına geliyor.

Bit 1 **EOC**: Regular channel end of conversion

This bit is set by hardware at the end of the conversion of a regular group of channels. It is cleared by software or by reading the ADC\_DR register.

0: Conversion not complete (EOCS=0), or sequence of conversions not complete (EOCS=1)  
1: Conversion complete (EOCS=0), or sequence of conversions complete (EOCS=1)

```
while(!(ADC1->SR & 0x00000002)); //EOC
```

- Daha sonra ADC\_CR registerinden okuma yapıyoruz.

## ADC regular data register (ADC\_DR)

Address offset: 0x4C

Reset value: 0x0000 0000

|            |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
|------------|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|
| 31         | 30 | 29 | 28 | 27 | 26 | 25 | 24 | 23 | 22 | 21 | 20 | 19 | 18 | 17 | 16 |
| Reserved   |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| 15         | 14 | 13 | 12 | 11 | 10 | 9  | 8  | 7  | 6  | 5  | 4  | 3  | 2  | 1  | 0  |
| DATA[15:0] |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| r          | r  | r  | r  | r  | r  | r  | r  | r  | r  | r  | r  | r  | r  | r  | r  |

Bits 31:16 Reserved, must be kept at reset value.

Bits 15:0 **DATA[15:0]**: Regular data

These bits are read-only. They contain the conversion result from the regular channels. The data are left- or right-aligned as shown in [Figure 48](#) and [Figure 49](#).

- Okunan değeri value değişkenine atıyoruz.

```
value=ADC1->DR;
```

```
40 uint8_t Read_ADC()
41 {
42     uint8_t value;
43
44     ADC1->CR2 |= 0x40000000;           //SWSTART
45     while(!(ADC1->SR & 0x00000002)); //EOC
46
47     value=ADC1->DR;                  //DATA
48     return value;
49 }
```

  

```
51 int main(void)
52 {
53     RCC_Config();
54     SystemCoreClockUpdate();
55     GPIO_Config();
56     ADC_Config();
57
58     while (1)
59     {
60         adc_value = Read_ADC();
61     }
62 }
```

- STMStudio programını açıp File kısmından Import variables tıklıyoruz ve çalışma dosyamızın debug içerisindeki elf. Uzantılı dosyayı seçiyoruz ve içerisinde okuyacağımız değişkeni import edip yandaki grafiğe ekliyoruz.
- A0'a bağladığımız potansiyometreyi çevirdikçe değer değişecektir.

| Variable Name | Address/Expression | Read Value |
|---------------|--------------------|------------|
| adc_value     | 0x200000a8         | 253        |

## 03\_02 ADC Interrupt

25 Aralık 2021 Cumartesi 00:56

## 03\_02 ADC Interrupt

HAL

## Konfigürasyon Kısmı



| Pin Name | Signal on Pin | GPIO output L | GPIO mode   | GPIO Pull-up/L   | Maximum out... | User Label | Modified                 |
|----------|---------------|---------------|-------------|------------------|----------------|------------|--------------------------|
| PA0-WKUP | ADC1_IN0      | n/a           | Analog mode | No pull-up an... | n/a            |            | <input type="checkbox"/> |
| PA1      | ADC2_IN1      | n/a           | Analog mode | No pull-up an... | n/a            |            | <input type="checkbox"/> |

- ADC1 için IN0, Temperature Sensor Channel ve Vrefint Channel seçimi yaptık.

- IN0
- IN1
- IN2
- IN3
- IN4
- IN5
- IN6
- IN7
- IN8
- IN9
- IN10
- IN11
- IN12
- IN13
- IN14
- IN15

Temperature Sensor Channel

Vrefint Channel

- Scan Mode ile Continuous Mode Enabled yapılır.

## ADC Settings

|                                      |                                                  |
|--------------------------------------|--------------------------------------------------|
| <u>Clock Prescaler</u>               | PCLK2 divided by 4                               |
| <u>Resolution</u>                    | 12 bits (15 ADC Clock cycles)                    |
| <u>Data Alignment</u>                | Right alignment                                  |
| <u>Scan Conversion Mode</u>          | Enabled                                          |
| <u>Continuous Conversion Mode</u>    | Enabled                                          |
| <u>Discontinuous Conversion Mode</u> | Disabled                                         |
| <u>DMA Continuous Requests</u>       | Disabled                                         |
| <u>End Of Conversion Selection</u>   | EOC flag at the end of single channel conversion |

- ADC1 için 3 kanal olduğundan her biri için rank işlemi yapılır.

|                                    |                                         |
|------------------------------------|-----------------------------------------|
| ADC-Regular_ConversionMode         |                                         |
| Number Of Conversion               | 3                                       |
| External Trigger Conversion Source | Regular Conversion launched by software |
| External Trigger Conversion Edge   | None                                    |
| Rank                               | 1                                       |
| Channel                            | Channel 0                               |
| Sampling Time                      | 56 Cycles                               |
| Rank                               | 2                                       |
| Channel                            | Channel Vrefint                         |
| Sampling Time                      | 56 Cycles                               |
| Rank                               | 3                                       |
| Channel                            | Channel Temperature Sensor              |
| Sampling Time                      | 56 Cycles                               |

- ADC2 için IN1 seçimi yapılır.
- ADC2 için tek kanal olduğundan Scan Mode seçimi yapılmaz.

|                               |                                                  |
|-------------------------------|--------------------------------------------------|
| ADC_Settings                  |                                                  |
| Clock Prescaler               | PCLK2 divided by 4                               |
| Resolution                    | 12 bits (15 ADC Clock cycles)                    |
| Data Alignment                | Right alignment                                  |
| Scan Conversion Mode          | Disabled                                         |
| Continuous Conversion Mode    | Enabled                                          |
| Discontinuous Conversion Mode | Disabled                                         |
| DMA Continuous Requests       | Disabled                                         |
| End Of Conversion Selection   | EOC flag at the end of single channel conversion |

|                                    |                                         |
|------------------------------------|-----------------------------------------|
| ADC-Regular_ConversionMode         |                                         |
| Number Of Conversion               | 1                                       |
| External Trigger Conversion Source | Regular Conversion launched by software |
| External Trigger Conversion Edge   | None                                    |
| Rank                               | 1                                       |
| Channel                            | Channel 1                               |
| Sampling Time                      | 56 Cycles                               |

- Interrupt kullanacağımızdan NVIC Settings kısmından Enabled yapılır.
- Sadece ADC1 değil ADC2 ve ADC3 için interrupts Enabled yapılmış olur.

| NVIC Interrupt Table           | Enabled                             | Preemption Priority | Sub Priority |
|--------------------------------|-------------------------------------|---------------------|--------------|
| ADC1, ADC2 and ADC3 interrupts | <input checked="" type="checkbox"/> | 0                   | 0            |

### Kod Kısmı

- ADC için Interrupt Mode kullanımı yapacağız.

```

96     *** Interrupt mode IO operation ***
97     =====
98     [...]
99     (+) Start the ADC peripheral using HAL_ADC_Start_IT()
100    (+) Use HAL_ADC_IRQHandler() called under ADC_IRQHandler() Interrupt subroutine
101    (+) At ADC end of conversion HAL_ADC_ConvCpltCallback() function is executed and user can
102        add his own code by customization of function pointer HAL_ADC_ConvCpltCallback
103    (+) In case of ADC Error, HAL_ADC_ErrorCallback() function is executed and user can
104        add his own code by customization of function pointer HAL_ADC_ErrorCallback
105    (+) Stop the ADC peripheral using HAL_ADC_Stop_IT()

```

```
1038= HAL_StatusTypeDef HAL_ADC_Start_IT(ADC_HandleTypeDef* hadc)
```

- Keseme olduğunda it.c dosyasındaki ADC\_IRQHandler fonksiyonuna gelir ve fonksiyon içerisindeki HAL\_ADC\_IRQHandler fonksiyonunu çalıştırır.
- Bizim main.c dosyasında hazır olarak yazılmıştır.

```

206= void ADC_IRQHandler(void)
207 {
208     /* USER CODE BEGIN ADC_IRQHandler_0 */
209
210     /* USER CODE END ADC_IRQHandler_0 */
211     HAL_ADC_IRQHandler(&hadc1);
212     HAL_ADC_IRQHandler(&hadc2);
213     /* USER CODE BEGIN ADC_IRQHandler_1 */
214
215     /* USER CODE END ADC_IRQHandler_1 */
216 }

```

- 1578.satırındaki fonksiyonu main.c dosyasında kullanarak Interrupt'a girdiğinde çalışacak.

```
1578= __weak void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* hadc)
```

- İçerisine önce hangi ADC'yi çalıştırıldığını öğrenmemiz gerekiyor. Bunun için \_HAL\_ADC\_GET\_FLAG kullanmamız gerekiyor.

129        (+) `_HAL_ADC_GET_FLAG: Get the selected ADC's flag status`

548 `#define __HAL_ADC_GET_FLAG(__HANDLE__, __FLAG__) (((__HANDLE__)->Instance->SR) & (__FLAG__)) == (__FLAG__)`

- Temperature için Datasheet kısmından aşağıdaki tabloları kullanıyoruz.

Temperature sensor characteristics

| Symbol                   | Parameter                                                      | Min | Typ     | Max     | Unit  |
|--------------------------|----------------------------------------------------------------|-----|---------|---------|-------|
| $T_L^{(1)}$              | $V_{SENSE}$ linearity with temperature                         | -   | $\pm 1$ | $\pm 2$ | °C    |
| Avg_Slope <sup>(1)</sup> | Average slope                                                  | -   | 2.5     | -       | mV/°C |
| $V_{25}^{(1)}$           | Voltage at 25 °C                                               | -   | 0.76    | -       | V     |
| $t_{START}^{(2)}$        | Startup time                                                   | -   | 6       | 10      | μs    |
| $T_{S\_temp}^{(2)}$      | ADC sampling time when reading the temperature (1 °C accuracy) | 10  | -       | -       | μs    |

Temperature sensor calibration values

| Symbol  | Parameter                                                          | Memory address            |
|---------|--------------------------------------------------------------------|---------------------------|
| TS_CAL1 | TS ADC raw data acquired at temperature of 30 °C, $V_{DDA}=3.3$ V  | 0x1FFF 7A2C - 0x1FFF 7A2D |
| TS_CAL2 | TS ADC raw data acquired at temperature of 110 °C, $V_{DDA}=3.3$ V | 0x1FFF 7A2E - 0x1FFF 7A2F |

- Aşağıdaki formül üzerinden hesaplıyoruz.

$$\text{Temperature (in } ^\circ\text{C)} = \{(V_{SENSE} - V_{25}) / \text{Avg\_Slope}\} + 25$$

- $V_{25}$  =  $V_{SENSE}$  value for 25° C
- Avg\_Slope = average slope of the temperature vs.  $V_{SENSE}$  curve (given in mV/°C or  $\mu$ V/°C)

```

23/* Private includes -----
24 /* USER CODE BEGIN Includes */
25 #define VREFIN_CAL ((uint16_t*)((uint32_t)0x1FFF7A2A))
26 #define V25 (float) 0.76
27 #define Avg_Slope (float) 0.0025
28
29 uint16_t adc1_value[3], adc2_value;
30 float Vadc1, Vadc2, Vsense, Vdda, temperature;
31 int count=0;
32 /* USER CODE END Includes */

121 /* USER CODE BEGIN 2 */
122 HAL_ADC_Start_IT(&hadc1);
123 HAL_ADC_Start_IT(&hadc2);
124 /* USER CODE END 2 */

```

```

65/* Private user code -----*/
66 /* USER CODE BEGIN 0 */
67 void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* hadc)
68 {
69     if(__HAL_ADC_GET_FLAG(&hadc1, ADC_FLAG_EOC) != RESET)
70     {
71         adc1_value[count]= HAL_ADC_GetValue(&hadc1);
72         count++; // count=0 => IN0, count=1 => Vrefint, count=2 => Temperature
73         if(count==3)
74         {
75             count=0;
76         }
77         Vdda= (float) 3.3 * (*VREFIN_CAL) / adc1_value[1];
78         Vadc1= Vdda * adc1_value[0] / 4095;
79
80         Vsense = Vdda * adc1_value[2] / 4095;
81         temperature= ((Vsense - V25) / Avg_Slope) + 25;
82     }
83     if(__HAL_ADC_GET_FLAG(&hadc2,ADC_FLAG_EOC) != RESET)
84     {
85         adc2_value=HAL_ADC_GetValue(&hadc2);
86         Vadc2= Vdda * adc2_value / 4095;
87     }
88 }
89 /* USER CODE END 0 */

```

| Variable Name | Address/Expression | Read Value |
|---------------|--------------------|------------|
| Vadc1         | 0x20000028         | 2.9433491  |
| Vadc2         | 0x2000002c         | 1.9205443  |
| Vdda          | 0x20000030         | 2.9363744  |
| Vsense        | 0x20000034         | 0.782018   |
| count         | 0x20000038         | 0          |
| temperature   | 0x2000003c         | 33.065987  |

04\_01 DAC Kullanımı

25 Aralık 2021 Cumartesi 00:57

## 04\_01 DAC Kullanımı

➤ HAL

## Konfigürasyon Kısmı



| Pin N... | Signal on... | GPIO out... | GPIO mo... | GPIO Pul...  | Maximu... | User Label | Modified                 |
|----------|--------------|-------------|------------|--------------|-----------|------------|--------------------------|
| PA4      | DAC_O...     | n/a         | Analog ... | No pull-u... | n/a       |            | <input type="checkbox"/> |

- DAC kısmından OUT1 seçiyoruz.

- ### OUT1 Configuration

- #### OUT2 Configuration

- External Trigger

- Output Buffer, gürültü engelleme; Trigger tetiklemedir. Biz bunları aynı şekilde bırakıyoruz.

## DAC Out1 Settings

Output Buffer Enable  
Trigger None

Kod Kısmı

- DAC için toplam iki kanal var.

```
18     [...]
19     *** DAC Channels ***
20     =====
21     [...]
22 STM32F4 devices integrate two 12-bit Digital Analog Converters
23
24 The 2 converters (i.e. channel1 & channel2)
25 can be used independently or simultaneously (dual mode):
26     (#) DAC channel1 with DAC_OUT1 (PA4) as output
27     (#) DAC channel2 with DAC_OUT2 (PA5) as output
```

- DAC kullanmak için iki modumuz var. Biri Polling diğeri DMA'dır.
- Biz Polling Mode kullanıyoruz.

```

110     *** Polling mode IO operation ***
111     =====
112     [...]
113         (+) Start the DAC peripheral using HAL_DAC_Start()
114         (+) To read the DAC last data output value, use the HAL_DAC_GetValue() function.
115         (+) Stop the DAC peripheral using HAL_DAC_Stop()

```

435 HAL\_StatusTypeDef HAL\_DAC\_Start(DAC\_HandleTypeDef \*hdac, uint32\_t Channel)

- DAC için değer verme yapacağımızdan GetValue değil SetValue kullanacağız. Çıkışın voltaj değeri ile oynamış oluyoruz.

```

759 /**
760  * @brief Set the specified data holding register value for DAC channel.
761  * @param hdac pointer to a DAC_HandleTypeDef structure that contains
762  *             the configuration information for the specified DAC.
763  * @param Channel The selected DAC channel.
764  *             This parameter can be one of the following values:
765  *             @arg DAC_CHANNEL_1: DAC Channel1 selected
766  *             @arg DAC_CHANNEL_2: DAC Channel2 selected
767  * @param Alignment Specifies the data alignment.
768  *             This parameter can be one of the following values:
769  *             @arg DAC_ALIGN_8B_R: 8bit right data alignment selected
770  *             @arg DAC_ALIGN_12B_L: 12bit left data alignment selected
771  *             @arg DAC_ALIGN_12B_R: 12bit right data alignment selected
772  * @param Data Data to be loaded in the selected data holding register.
773  * @retval HAL status
774 */
775 HAL_StatusTypeDef HAL_DAC_SetValue(DAC_HandleTypeDef *hdac, uint32_t Channel, uint32_t Alignment, uint32_t Data)

```

- Data kısmına 12 bit çalıştığımızdan 0-4095 arasında değer yazabiliriz.

```

23 /* Private includes -----
24 /* USER CODE BEGIN Includes */
25 int i=0;
26 /* USER CODE END Includes */

```

```

91  /* USER CODE BEGIN 2 */
92 HAL_DAC_Start(&hdac, DAC_CHANNEL_1);
93  /* USER CODE END 2 */

```

```

95  /* Infinite loop */
96  /* USER CODE BEGIN WHILE */
97  while (1)
98  {
99      /* USER CODE END WHILE */
100
101     /* USER CODE BEGIN 3 */
102     do
103     {
104         HAL_DAC_SetValue(&hdac, DAC_CHANNEL_1, DAC_ALIGN_12B_R, i);
105         HAL_Delay(1);
106         i++;
107     }
108     while(i<4096);
109     i=0;
110 }
111 /* USER CODE END 3 */

```

## ➤ REGISTER

### Konfigürasyon Kısmı

- RCC için 5.satırda 0x00030000 yerine 0x00010000 yazdık.

HSERDY kısmı sadece okunabilir olduğundan o biti 1 yapmıyoruz.

```

3 void RCC_Config(void)
4 {
5     RCC->CR |= 0x00010000; //HSEON
6     while(!(RCC->CR & 0x00020000)); //HSERDY
7     RCC->CR |= 0x00080000; //CSSON
8     RCC->CFGGR = 0x00000000;
9     RCC->PLLCFGR |= 0x00400000; //PLLSRC
10    RCC->PLLCFGR |= 0x00000004; //PLLM 4
11    RCC->PLLCFGR |= 0x00002A00; //PLLN 168
12    RCC->PLLCFGR |= 0x00000000; //PLLP 2
13    RCC->CR |= 0x01000000; //PLLON
14    while(!(RCC->CR & 0x02000000)); //PLLRDY
15    RCC->CFGGR |= 0x00000001; //SW
16    while(!(RCC->CR & 0x00000001)); //SWS
17 }

```

- ADC'nin clock hattı APB1'e gitiyor.



### 7.3.13 RCC APB1 peripheral clock enable register (RCC\_APB1ENR)

Address offset: 0x40

Reset value: 0x0000 0000

Access: no wait state, word, half-word and byte access.

| 31       | 30      | 29       | 28      | 27            | 26       | 25       | 24            | 23      | 22      | 21      | 20       | 19       | 18         | 17         | 16            |
|----------|---------|----------|---------|---------------|----------|----------|---------------|---------|---------|---------|----------|----------|------------|------------|---------------|
| Reserved |         | DAC EN   | PWR EN  | Reser-<br>ved | CAN2 EN  | CAN1 EN  | Reser-<br>ved | I2C3 EN | I2C2 EN | I2C1 EN | UART5 EN | UART4 EN | USART 3 EN | USART 2 EN | Reser-<br>ved |
|          |         | rw       | rw      |               | rw       | rw       |               | rw      | rw      | rw      | rw       | rw       | rw         | rw         |               |
| 15       | 14      | 13       | 12      | 11            | 10       | 9        | 8             | 7       | 6       | 5       | 4        | 3        | 2          | 1          | 0             |
| SPI3 EN  | SPI2 EN | Reserved | WWDG EN | Reserved      | TIM14 EN | TIM13 EN | TIM12 EN      | TIM7 EN | TIM6 EN | TIM5 EN | TIM4 EN  | TIM3 EN  | TIM2 EN    |            |               |
| rw       | rw      |          | rw      |               | rw       | rw       | rw            | rw      | rw      | rw      | rw       | rw       | rw         | rw         |               |

- 29.pini 1 yapıyoruz.

Bit 29 **DACEN**: DAC interface clock enable

Set and cleared by software.

0: DAC interface clock disabled

1: DAC interface clock enable

```
RCC->APB1ENR |= 0x20000000;
```

//DACEN

## DAC control register (DAC\_CR)

Address offset: 0x00

Reset value: 0x0000 0000

- DAC işlemi için channel1 kullanacağımızdan 0.biti aktif etmemiz gerekiyor.

Bit 0 EN1: DAC channel1 enable

This bit is set and cleared by software to enable/disable DAC channel1.

0: DAC channel1 disabled

1: DAC channel1 enabled

DAC->CR |= 0x00000001;

- Yazılımsal tetikleme kullanmadığımızdan 0. biti 0 yapıyoruz.

## DAC software trigger register (DAC\_SWTRIGR)

Address offset: 0x04

Reset value: 0x0000 0000

Bit 0 SWTRIG1: DAC channel1 software trigger

This bit is set and cleared by software to enable/disable the software trigger.

0: Software trigger disabled

### 1: Software trigger enabled

**Note:** This bit is cleared by hardware (one APB1 clock cycle later) once the DAC\_DHR1 register value has been loaded into the DAC\_DOR1 register.

```
DAC->SWTRIGR |= 0x00000000;
```

- Channel1 için 12 bit sağa ya da sola veya 8 bit sağa kaydedebiliriz.

## DAC channel1 12-bit right-aligned data holding register (DAC\_DHR12R1)

Address offset: 0x08

Reset value: 0x0000 0000

## DAC channel1 12-bit left aligned data holding register (DAC\_DHR12L1)

Address offset: 0x0C

Reset value: 0x0000 0000

|                |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
|----------------|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|
| 31             | 30 | 29 | 28 | 27 | 26 | 25 | 24 | 23 | 22 | 21 | 20 | 19 | 18 | 17 | 16 |
| Reserved       |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| 15             | 14 | 13 | 12 | 11 | 10 | 9  | 8  | 7  | 6  | 5  | 4  | 3  | 2  | 1  | 0  |
| DACC1DHR[11:0] |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| rw             | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw |

## DAC channel1 8-bit right aligned data holding register (DAC\_DHR8R1)

Address offset: 0x10

Reset value: 0x0000 0000

|               |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
|---------------|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|
| 31            | 30 | 29 | 28 | 27 | 26 | 25 | 24 | 23 | 22 | 21 | 20 | 19 | 18 | 17 | 16 |
| Reserved      |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| 15            | 14 | 13 | 12 | 11 | 10 | 9  | 8  | 7  | 6  | 5  | 4  | 3  | 2  | 1  | 0  |
| DACC1DHR[7:0] |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| rw            | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw |

- 12 bit sağa kaydedeceğiz.

Bits 11:0 DACC1DHR[11:0]: DAC channel1 12-bit right-aligned data

These bits are written by software which specifies 12-bit data for DAC channel1.

## RCC AHB1 peripheral clock enable register (RCC\_AHB1ENR)

Address offset: 0x30

Reset value: 0x0010 0000

Access: no wait state, word, half-word and byte access.

|               |                         |             |                     |                    |                    |              |          |            |             |             |                  |         |               |             |             |             |  |
|---------------|-------------------------|-------------|---------------------|--------------------|--------------------|--------------|----------|------------|-------------|-------------|------------------|---------|---------------|-------------|-------------|-------------|--|
| 31            | 30                      | 29          | 28                  | 27                 | 26                 | 25           | 24       | 23         | 22          | 21          | 20               | 19      | 18            | 17          | 16          |             |  |
| Reser-<br>ved | OTGH<br>S<br>ULPIE<br>N | OTGH<br>SEN | ETHM<br>ACPTP<br>EN | ETHM<br>ACRXE<br>N | ETHM<br>ACTXE<br>N | ETHMA<br>CEN | Reserved |            | DMA2E<br>N  | DMA1E<br>N  | CCMDAT<br>ARAMEN | Res.    | BKPSR<br>AMEN | Reserved    |             |             |  |
|               | rw                      | rw          | rw                  | rw                 | rw                 | rw           |          |            | rw          | rw          |                  |         | rw            |             |             |             |  |
| 15            | 14                      | 13          | 12                  | 11                 | 10                 | 9            | Reserved | 8          | 7           | 6           | 5                | 4       | 3             | 2           | 1           |             |  |
| Reserved      |                         |             |                     | CRCE<br>N          | Reserved           |              |          | GPIOE<br>N | GPIOH<br>EN | GPIOG<br>EN | GPIOF<br>EN      | GPIOEEN | GPIOD<br>EN   | GPIOC<br>EN | GPIO<br>BEN | GPIO<br>AEN |  |
|               |                         |             |                     | rw                 |                    |              |          | rw         | rw          | rw          | rw               | rw      | rw            | rw          | rw          |             |  |

- DAC1 A4 pini olduğundan portunu sadece A portunu aktif ediyoruz.

RCC->[AHB1ENR](#) |= 0x00000001;

- RCC ve DAC1 için yazdığımız fonksiyonlar aşağıdaki gibidir.

```

10 void RCC_Config(void)
11 {
12     RCC->CR |= 0x00010000; //HSEON
13     while(!(RCC->CR & 0x00020000)); //HSERDY
14     RCC->CR |= 0x00080000; //CSSON
15     RCC->CFGR = 0x00000000;
16     RCC->PLLCFGR |= 0x00400000; //PLLSRC
17     RCC->PLLCFGR |= 0x00000004; //PLLM 4
18     RCC->PLLCFGR |= 0x00002A00; //PLLN 168
19     RCC->PLLCFGR |= 0x00000000; //PLLP 2
20     RCC->CR |= 0x01000000; //PLLON
21     while(!(RCC->CR & 0x02000000)); //PLLRDY
22     RCC->CFGR |= 0x00000001; //SW
23     while(!(RCC->CR & 0x00000001)); //SWS
24 }

26 void DAC1_Config(void)
27 {
28     RCC->APB1ENR |= 0x20000000; //DACEEN
29     RCC->AHB1ENR |= 0x00000001; //A clock enable
30
31     DAC->CR |= 0x00000001; //DAC channel1 enabled
32     DAC->SWTRIGR |= 0x00000000; //Software trigger disabled
33     DAC->DHR12R1 |= 0x00000000; //DAC channel1 12-bit right-aligned data
34 }

```

## Kod Kısmı

```

3 int i=0;
4
5 void delay(uint32_t time)
6 {
7     while(time--);
8 }

```

- Her i değeri arttığında i değeri kadar kaydetmiş oluyor.
- A4 pinine led bağladığımızda i değeri arttıkça voltaj değeri artacağından parlaklık artar.
- 4095 olduğunda yani 0x0000FFF olduğunda tam parlaklıktta yanar.

```

36 int main(void)
37 {
38     RCC_Config();
39     SystemCoreClockUpdate();
40     DAC1_Config();
41
42     while (1)
43     {
44         for(;i<4096; i++)
45         {
46             DAC->DHR12R1 |= i;
47             delay(16800);
48         }
49         i=0;
50     }
51 }

```

# 04\_02 ADC Değeri ile DAC Kontrolü

25 Aralık 2021 Cumartesi 00:57

## 04\_02 ADC Değeri ile DAC Kontrolü

➤ HAL

### Konfigürasyon Kısmı



| Pin Na... | Signal on ... | GPIO outp... | GPIO mode   | GPIO Pull...   | Maximum ... | User Label | Modified                 |
|-----------|---------------|--------------|-------------|----------------|-------------|------------|--------------------------|
| PA0-WK... | ADC1_IN0      | n/a          | Analog m... | No pull-up ... | n/a         |            | <input type="checkbox"/> |
| Pin Na... | Signal on Pin | GPIO out...  | GPIO mode   | GPIO P...      | Maximum...  | User L...  | Modified                 |
| PA4       | DAC_OUT1      | n/a          | Analog m... | No pull-...    | n/a         |            | <input type="checkbox"/> |

- ADC1 için IN0, DAC için OUT1 seçilir.

#### ADC\_Settings

|                               |                                                  |
|-------------------------------|--------------------------------------------------|
| Clock Prescaler               | PCLK2 divided by 2                               |
| Resolution                    | 12 bits (15 ADC Clock cycles)                    |
| Data Alignment                | Right alignment                                  |
| Scan Conversion Mode          | Disabled                                         |
| Continuous Conversion Mode    | Enabled                                          |
| Discontinuous Conversion Mode | Disabled                                         |
| DMA Continuous Requests       | Disabled                                         |
| End Of Conversion Selection   | EOC flag at the end of single channel conversion |

#### DAC Out1 Settings

|               |        |
|---------------|--------|
| Output Buffer | Enable |
| Trigger       | None   |

### Kod Kısmı

```

23/* Private includes -----
24 /* USER CODE BEGIN Includes */
25 uint16_t adc_value, dac_value;
26 /* USER CODE END Includes */
• 16 bitlik okuma yapan fonksiyon yazdık.
• Geri dönüşlü fonksiyon olduğundan Return ile değeri geri
  döndürüyor.
60/* Private user code -----
61 /* USER CODE BEGIN 0 */
62uint16_t Read_ADC()
63 {
64     HAL_ADC_Start(&hadc1);
65
66     if(HAL_ADC_PollForConversion(&hadc1, 100000) == HAL_OK)
67     {
68         adc_value = HAL_ADC_GetValue(&hadc1);
69
70     }
71     HAL_ADC_Stop(&hadc1);
72
73     return adc_value;
74 }
75 /* USER CODE END 0 */

107 /* USER CODE BEGIN 2 */
108 HAL_DAC_Start(&hdac, DAC_CHANNEL_1);
109 /* USER CODE END 2 */

111 /* Infinite loop */
112 /* USER CODE BEGIN WHILE */
113 while (1)
114 {
115     /* USER CODE END WHILE */
116
117     /* USER CODE BEGIN 3 */
118     dac_value=Read_ADC();
119     HAL_DAC_SetValue(&hdac, DAC_CHANNEL_1, DAC_ALIGN_12B_R, dac_value);
120 }
121 /* USER CODE END 3 */
122 }

```

## 05\_01 DMA ile ADC Değer Okuma

25 Aralık 2021 Cumartesi 00:58

## 05\_01 DMA ile ADC Değer Okuma

➤ HAL

## Konfigürasyon Kısmı



| Pin Name | Signal on Pin | GPIO output | GPIO mode   | GPIO Pull-up | Maximum output | User Label | Modified                 |
|----------|---------------|-------------|-------------|--------------|----------------|------------|--------------------------|
| PA0-WKUP | ADC1_IN0      | n/a         | Analog mode | No pull-up   | n/a            |            | <input type="checkbox"/> |
| PA1      | ADC1_IN1      | n/a         | Analog mode | No pull-up   | n/a            |            | <input type="checkbox"/> |

## ADC\_Settings

|                               |                                                  |
|-------------------------------|--------------------------------------------------|
| Clock Prescaler               | PCLK2 divided by 4                               |
| Resolution                    | 12 bits (15 ADC Clock cycles)                    |
| Data Alignment                | Right alignment                                  |
| Scan Conversion Mode          | Enabled                                          |
| Continuous Conversion Mode    | Enabled                                          |
| Discontinuous Conversion Mode | Disabled                                         |
| DMA Continuous Requests       | Enabled                                          |
| End Of Conversion Selection   | EOC flag at the end of single channel conversion |

- DMA ayarları için Add diyoruz. Ardından Stream, Direction ve Priority için seçenekleri seçiyoruz. Daha sonra Direction kısmından seçtiğimiz Peripheral ve Memory için adres değişikliği için Increment Adress için işaretliyoruz. Ardından gönderilecek data boyutu belirliyoruz. 12bit çalıştığımızdan 16 bitlik olan Half Word seçimi yapıyoruz.

| DMA Request | Stream        | Direction            | Priority  |
|-------------|---------------|----------------------|-----------|
| ADC1        | DMA2 Stream 0 | Peripheral To Memory | Very High |

DMA Request Settings

|          |                                         |                                            |                                                                                        |
|----------|-----------------------------------------|--------------------------------------------|----------------------------------------------------------------------------------------|
| Mode     | <input type="button" value="Circular"/> | <input type="checkbox"/> Increment Address | <input checked="" type="checkbox"/> Memory                                             |
| Use Fifo | <input type="checkbox"/>                | Threshold                                  | <input type="button" value="Half Word"/><br><input type="button" value="Half Word"/>   |
|          |                                         | Data Width                                 | <input type="button" value="Burst Size"/><br><input type="button" value="Burst Size"/> |

## Kod Kısmı

```

107      *** DMA mode IO operation ***
108      =====
109      [...]
110      (+) Start the ADC peripheral using HAL_ADC_Start_DMA(), at this stage the user specify the length
111          of data to be transferred at each end of conversion
112      (+) At The end of data transfer by HAL_ADC_ConvCpltCallback() function is executed and user can
113          add his own code by customization of function pointer HAL_ADC_ConvCpltCallback
114      (+) In case of transfer Error, HAL_ADC_ErrorCallback() function is executed and user can
115          add his own code by customization of function pointer HAL_ADC_ErrorCallback
116      (+) Stop the ADC peripheral using HAL_ADC_Stop_DMA()

```

- pData kısmına veriyi tutacağımız değişkeni yazıyoruz. Bizim değişken 16 bit olarak tanımladığımızdan dönüştürmemiz gerekiyor.
- Length ile kaç tane veri okuyacağımızı yazıyoruz.

```

1354 /**
1355  * @brief Enables ADC DMA request after last transfer (Single-ADC mode) and enables ADC
1356  * peripheral
1357  * @param hadc pointer to a ADC_HandleTypeDef structure that contains
1358  * the configuration information for the specified ADC.
1359  * @param pData The destination Buffer address.
1360  * @param Length The length of data to be transferred from ADC peripheral to memory.
1361  * @retval HAL status
1362 */
1363 HAL_StatusTypeDef HAL_ADC_Start_DMA(ADC_HandleTypeDef* hadc, uint32_t* pData, uint32_t Length)

```

```

46 /* USER CODE BEGIN PV */
47 uint16_t adc_value[2];
48 /* USER CODE END PV */

```

```

94 /* USER CODE BEGIN 2 */
95 HAL_ADC_Start_DMA(&hadc1, (uint32_t*)adc_value, 2);
96 /* USER CODE END 2 */

```

## ➤ REGISTER

## Konfigürasyon Kısmı

- RCC için eklemeler yaptık.

## RCC clock configuration register (RCC\_CFGR)

Address offset: 0x08

Reset value: 0x0000 0000

Access:  $0 \leq \text{wait state} \leq 2$ , word, half-word and byte access

1 or 2 wait states inserted only if the access occurs during a clock source switch.

| 31         | 30 | 29            | 28         | 27            | 26 | 25       | 24        | 23   | 22 | 21          | 20   | 19   | 18  | 17  | 16 |
|------------|----|---------------|------------|---------------|----|----------|-----------|------|----|-------------|------|------|-----|-----|----|
| MCO2       |    | MCO2 PRE[2:0] |            | MCO1 PRE[2:0] |    | I2SSC R  |           | MCO1 |    | RTCPRE[4:0] |      |      |     |     |    |
| rw         |    | rw            | rw         | rw            | rw | rw       | rw        | rw   | rw | rw          | rw   | rw   | rw  | rw  | rw |
| 15         | 14 | 13            | 12         | 11            | 10 | 9        | 8         | 7    | 6  | 5           | 4    | 3    | 2   | 1   | 0  |
| PPRE2[2:0] |    |               | PPRE1[2:0] |               |    | Reserved | HPRE[3:0] |      |    |             | SWS1 | SWS0 | SW1 | SW0 |    |
| rw         | rw | rw            | rw         | rw            | rw |          | rw        | rw   | rw | rw          | r    | r    | rw  | rw  |    |

- AHB, APB1 ve APB2 için clock hızını böldük.

Bits 7:4 **HPRE**: AHB prescaler

Set and cleared by software to control AHB clock division factor.

**Caution:** The clocks are divided with the new prescaler factor from 1 to 16 AHB cycles after HPRE write.

**Caution:** The AHB clock frequency must be at least 25 MHz when the Ethernet is used.

0xxx: system clock not divided

1000: system clock divided by 2

1001: system clock divided by 4

1010: system clock divided by 8

1011: system clock divided by 16

1100: system clock divided by 64

1101: system clock divided by 128

1110: system clock divided by 256

1111: system clock divided by 512

RCC->**CFGR** |= 0x00000000;

Bits 12:10 **PPRE1**: APB Low speed prescaler (APB1)

Set and cleared by software to control APB low-speed clock division factor.

**Caution:** The software has to set these bits correctly not to exceed 42 MHz on this domain.

The clocks are divided with the new prescaler factor from 1 to 16 AHB cycles after PPRE1 write.

0xx: AHB clock not divided

100: AHB clock divided by 2

101: AHB clock divided by 4

110: AHB clock divided by 8

111: AHB clock divided by 16

RCC->**CFGR** |= 0x00001400;

Bits 15:13 **PPRE2**: APB high-speed prescaler (APB2)

Set and cleared by software to control APB high-speed clock division factor.

**Caution:** The software has to set these bits correctly not to exceed 84 MHz on this domain.

The clocks are divided with the new prescaler factor from 1 to 16 AHB cycles after PPRE2 write.

0xx: AHB clock not divided

100: AHB clock divided by 2

101: AHB clock divided by 4

110: AHB clock divided by 8

111: AHB clock divided by 16

RCC->**CFGR** |= 0x00008000;

## RCC clock interrupt register (RCC\_CIR)

Address offset: 0x0C

Reset value: 0x0000 0000

Access: no wait state, word, half-word and byte access

| 31       | 30          | 29        | 28        | 27        | 26        | 25        | 24       | 23       | 22         | 21         | 20       | 19       | 18       | 17       | 16       |
|----------|-------------|-----------|-----------|-----------|-----------|-----------|----------|----------|------------|------------|----------|----------|----------|----------|----------|
| Reserved |             |           |           |           |           | CSSC      | Reserved |          |            | PLL2S RDYC | PLL RDYC | HSE RDYC | HSI RDYC | LSE RDYC | LSI RDYC |
| 15       | 14          | 13        | 12        | 11        | 10        | 9         | 8        | 7        | 6          | 5          | 4        | 3        | 2        | 1        | 0        |
| Reserved | PLL2S RDYIE | PLL RDYIE | HSE RDYIE | HSI RDYIE | LSE RDYIE | LSI RDYIE | CSSF     | Reserved | PLL2S RDYF | PLL RDYF   | HSE RDYF | HSI RDYF | LSE RDYF | LSI RDYF |          |
|          | rw          | rw        | rw        | rw        | rw        | rw        | r        |          | r          | r          | r        | r        | r        | r        | r        |

- Kaldırılan bayrakları temizliyoruz.

Bit 19 **HSERDYC**: HSE ready interrupt clear

This bit is set by software to clear the HSERDYF flag.

0: No effect

1: HSERDYF cleared

```
RCC->CIR |= 0x00080000;
```

Bit 23 **CSSC**: Clock security system interrupt clear

This bit is set by software to clear the CSSF flag.

0: No effect

1: Clear CSSF flag

```
RCC->CIR |= 0x00800000;
```

- RCC için yazdığımız fonksiyon aşağıdaki gibidir.

```
3 void RCC_Config(void)
4 {
5     RCC->CR |= 0x00010000; //HSEON
6     while(!(RCC->CR & 0x00020000)); //HSERDY
7     RCC->CR |= 0x00080000; //CSSON
8     RCC->CFGGR = 0x00000000;
9     RCC->PLLCFGR |= 0x00400000; //PLLSRC
10    RCC->PLLCFGR |= 0x00000004; //PLLM 4
11    RCC->PLLCFGR |= 0x00002A00; //PLLN 168
12    RCC->PLLCFGR |= 0x00000000; //PLLP 2
13    RCC->CR |= 0x01000000; //PLLON
14    while(!(RCC->CR & 0x02000000)); //PLLRDY
15    RCC->CFGGR |= 0x00000001; //SW
16    while(!(RCC->CR & 0x00000001)); //SWS
17    RCC->CFGGR |= 0x00000000; //HPRE AHB 1
18    RCC->CFGGR |= 0x00001400; //PPRE1 APB1 4
19    RCC->CFGGR |= 0x00008000; //PPRE2 APB2 2
20    RCC->CIR |= 0x00080000; //HSERDYC
21    RCC->CIR |= 0x00800000; //CSSC
22 }
```

- GPIO için yazdığımız fonksiyon aşağıdaki gibidir.

```
27 void GPIO_Config(void)
28 {
29     RCC->AHB1ENR |= 0x00000001; //A clock enable
30
31     GPIOA->MODER |= 0x00000003; //PA0 Analog mode
32     GPIOA->OSPEEDR |= 0x00000003; //Very high speed
33 }
```

## ADC control register 1 (ADC\_CR1)

Address offset: 0x04

Reset value: 0x0000 0000

|              |    |    |              |            |       |            |      |        |        |          |            |    |    |    |    |
|--------------|----|----|--------------|------------|-------|------------|------|--------|--------|----------|------------|----|----|----|----|
| 31           | 30 | 29 | 28           | 27         | 26    | 25         | 24   | 23     | 22     | 21       | 20         | 19 | 18 | 17 | 16 |
| Reserved     |    |    |              |            | OVRIE | RES        |      | AWDEN  | JAWDEN | Reserved |            |    |    |    |    |
|              |    |    |              |            | rw    | rw         | rw   | rw     | rw     |          |            |    |    |    |    |
| 15           | 14 | 13 | 12           | 11         | 10    | 9          | 8    | 7      | 6      | 5        | 4          | 3  | 2  | 1  | 0  |
| DISCNUM[2:0] |    |    | JDISCN<br>EN | DISC<br>EN | JAUTO | AWDSG<br>L | SCAN | JEOCIE | AWDIE  | EOCIE    | AWDCH[4:0] |    |    |    |    |
| rw           | rw | rw | rw           | rw         | rw    | rw         | rw   | rw     | rw     | rw       | rw         | rw | rw | rw | rw |

- Çoklu ADC okuması yapılacaksça scan yani tarama aktif edilmesi gerekir.

Bit 8 **SCAN**: Scan mode

This bit is set and cleared by software to enable/disable the Scan mode. In Scan mode, the inputs selected through the ADC\_SQRx or ADC\_JSQRx registers are converted.

0: Scan mode disabled

1: Scan mode enabled

Note: An EOC interrupt is generated if the EOCIE bit is set:

- At the end of each regular group sequence if the EOCS bit is cleared to 0
- At the end of each regular channel conversion if the EOCS bit is set to 1

Note: A JEOC interrupt is generated only on the end of conversion of the last channel if the JEOCIE bit is set.

ADC1->**CR1** |= 1 << 8;

- Çözünürlük için 12-bit kullanıyoruz.

Bits 25:24 **RES[1:0]**: Resolution

These bits are written by software to select the resolution of the conversion.

00: 12-bit (15 ADCCLK cycles)

01: 10-bit (13 ADCCLK cycles)

10: 8-bit (11 ADCCLK cycles)

11: 6-bit (9 ADCCLK cycles)

ADC1->**CR1** |= 0 << 24;

## ADC control register 2 (ADC\_CR2)

Address offset: 0x08

Reset value: 0x0000 0000

|          |             |    |       |       |             |     |     |          |          |              |        |    |              |      |      |  |
|----------|-------------|----|-------|-------|-------------|-----|-----|----------|----------|--------------|--------|----|--------------|------|------|--|
| 31       | 30          | 29 | 28    | 27    | 26          | 25  | 24  | 23       | 22       | 21           | 20     | 19 | 18           | 17   | 16   |  |
| reserved | SWST<br>ART |    | EXTEN |       | EXTSEL[3:0] |     |     |          | reserved | JSWST<br>ART | JEXTEN |    | JEXTSEL[3:0] |      |      |  |
|          | rw          | rw | rw    | rw    | rw          | rw  | rw  | rw       |          | rw           | rw     | rw | rw           | rw   | rw   |  |
| 15       | 14          | 13 | 12    | 11    | 10          | 9   | 8   | 7        | 6        | 5            | 4      | 3  | 2            | 1    | 0    |  |
| reserved |             |    |       | ALIGN | EOCS        | DDS | DMA | Reserved |          |              |        |    |              | CONT | ADON |  |
|          |             |    |       | rw    | rw          | rw  | rw  |          |          |              |        |    |              | rw   | rw   |  |

- ADC aktif etmek için ilk bit 1 yapıldı.

Bit 0 **ADON**: A/D Converter ON / OFF

This bit is set and cleared by software.

Note: 0: Disable ADC conversion and go to power down mode

1: Enable ADC

- Sürekli okuma yapmasını istiyoruz.

Bit 1 **CONT**: Continuous conversion

This bit is set and cleared by software. If it is set, conversion takes place continuously until it is cleared.

0: Single conversion mode

1: Continuous conversion mode

- DMA mod ile çalışacağız.

Bit 8 **DMA**: Direct memory access mode (for single ADC mode)

This bit is set and cleared by software. Refer to the DMA controller chapter for more details.

0: DMA mode disabled

1: DMA mode enabled

- DMA'nın sürekli çalışması için 1 yapıyorum.

Bit 9 **DDS**: DMA disable selection (for single ADC mode)

This bit is set and cleared by software.

0: No new DMA request is issued after the last transfer (as configured in the DMA controller)

1: DMA requests are issued as long as data are converted and DMA=1

- 10.bit 1 yapılır.

Bit 10 **EOCS**: End of conversion selection

This bit is set and cleared by software.

0: The EOC bit is set at the end of each sequence of regular conversions. Overrun detection is enabled only if DMA=1.

1: The EOC bit is set at the end of each regular conversion. Overrun detection is enabled.

- Çevrim başlaması için 30.biti 1 yapıyoruz.

Bit 30 **SWSTART**: Start conversion of regular channels

This bit is set by software to start conversion and cleared by hardware as soon as the conversion starts.

0: Reset state

1: Starts conversion of regular channels

*Note: This bit can be set only when ADON = 1 otherwise no conversion is launched.*

ADC1->**CR2** |= 1 << 0;

ADC1->**CR2** |= 1 << 1;

ADC1->**CR2** |= 1 << 8;

ADC1->**CR2** |= 1 << 9;

ADC1->**CR2** |= 1 << 10;

ADC1->**CR2** |= 1 << 30;

- Kaç farklı kanaldan çevrim yapacağımızı belirtmemiz gerekiyor. Biz sadece A portun 0.bitinden yani 1 kanal'dan okuma yapacağım.

### ADC regular sequence register 1 (ADC\_SQR1)

Address offset: 0x2C

Reset value: 0x0000 0000

|          |           |    |    |    |    |    |           |    |    |        |    |           |           |    |    |
|----------|-----------|----|----|----|----|----|-----------|----|----|--------|----|-----------|-----------|----|----|
| 31       | 30        | 29 | 28 | 27 | 26 | 25 | 24        | 23 | 22 | 21     | 20 | 19        | 18        | 17 | 16 |
| Reserved |           |    |    |    |    |    |           |    |    | L[3:0] |    | SQ16[4:1] |           |    |    |
| 15       | 14        | 13 | 12 | 11 | 10 | 9  | 8         | 7  | 6  | 5      | 4  | 3         | 2         | 1  | 0  |
| SQ16_0   | SQ15[4:0] |    |    |    |    |    | SQ14[4:0] |    |    |        |    |           | SQ13[4:0] |    |    |
| rw       | rw        | rw | rw | rw | rw | rw | rw        |    |    |        |    | rw        | rw        | rw | rw |

### ADC regular sequence register 2 (ADC\_SQR2)

Address offset: 0x30

Reset value: 0x0000 0000

|          |          |           |    |    |    |    |           |    |    |    |    |           |          |    |    |
|----------|----------|-----------|----|----|----|----|-----------|----|----|----|----|-----------|----------|----|----|
| 31       | 30       | 29        | 28 | 27 | 26 | 25 | 24        | 23 | 22 | 21 | 20 | 19        | 18       | 17 | 16 |
| Reserved |          | SQ12[4:0] |    |    |    |    | SQ11[4:0] |    |    |    |    | SQ10[4:1] |          |    |    |
| 15       | 14       | 13        | 12 | 11 | 10 | 9  | 8         | 7  | 6  | 5  | 4  | 3         | 2        | 1  | 0  |
| SQ10_0   | SQ9[4:0] |           |    |    |    |    | SQ8[4:0]  |    |    |    |    |           | SQ7[4:0] |    |    |
| rw       | rw       | rw        | rw | rw | rw | rw | rw        | rw | rw | rw | rw | rw        | rw       | rw | rw |

### ADC regular sequence register 3 (ADC\_SQR3)

Address offset: 0x34

Reset value: 0x0000 0000

31 30 29 28 27 26 25 24 23 22 21 20 19 18 17 16

## ADC regular sequence register 3 (ADC\_SQR3)

Address offset: 0x34

Reset value: 0x0000 0000

|          |          |    |    |    |    |          |    |    |    |    |          |    |    |    |    |
|----------|----------|----|----|----|----|----------|----|----|----|----|----------|----|----|----|----|
| 31       | 30       | 29 | 28 | 27 | 26 | 25       | 24 | 23 | 22 | 21 | 20       | 19 | 18 | 17 | 16 |
| Reserved | SQ6[4:0] |    |    |    |    | SQ5[4:0] |    |    |    |    | SQ4[4:1] |    |    |    |    |
|          | rw       | rw | rw | rw | rw | rw       | rw | rw | rw | rw | rw       | rw | rw | rw | rw |
| 15       | 14       | 13 | 12 | 11 | 10 | 9        | 8  | 7  | 6  | 5  | 4        | 3  | 2  | 1  | 0  |
| SQ4_0    | SQ3[4:0] |    |    |    |    | SQ2[4:0] |    |    |    |    | SQ1[4:0] |    |    |    |    |
|          | rw       | rw | rw | rw | rw | rw       | rw | rw | rw | rw | rw       | rw | rw | rw | rw |

- SQR1 için 20.biti 0 yaparak 1 tane kanal olduğunu belirtiyoruz.

Bits 23:20 L[3:0]: Regular channel sequence length

These bits are written by software to define the total number of conversions in the regular channel conversion sequence.

0000: 1 conversion

0001: 2 conversions

...

1111: 16 conversions

ADC1->SQR1 |= 0 << 20;

- SQR3 için SQ1 kısmına kanal numarasını yazıyoruz.

Bits 4:0 SQ1[4:0]: 1st conversion in regular sequence

ADC1->SQR3 |= 0 << 0;

## ADC common control register (ADC\_CCR)

Address offset: 0x04 (this offset address is relative to ADC1 base address + 0x300)

Reset value: 0x0000 0000

|          |     |      |            |    |    |    |    |          |       |          |    |            |    |        |    |
|----------|-----|------|------------|----|----|----|----|----------|-------|----------|----|------------|----|--------|----|
| 31       | 30  | 29   | 28         | 27 | 26 | 25 | 24 | 23       | 22    | 21       | 20 | 19         | 18 | 17     | 16 |
| Reserved |     |      |            |    |    |    |    | TSVREFE  | VBATE | Reserved |    |            |    | ADCPRE |    |
|          |     |      |            |    |    |    |    | rw       | rw    |          |    |            |    | rw     | rw |
| 15       | 14  | 13   | 12         | 11 | 10 | 9  | 8  | 7        | 6     | 5        | 4  | 3          | 2  | 1      | 0  |
| DMA[1:0] | DDS | Res. | DELAY[3:0] |    |    |    |    | Reserved |       |          |    | MULTI[4:0] |    |        |    |
| rw       | rw  | rw   | rw         | rw | rw | rw | rw |          |       |          |    | rw         | rw | rw     | rw |

Bits 17:16 ADCPRE: ADC prescaler

Set and cleared by software to select the frequency of the clock to the ADC. The clock is common for all the ADCs.

Note: 00: PCLK2 divided by 2

01: PCLK2 divided by 4

10: PCLK2 divided by 6

11: PCLK2 divided by 8

ADC->CCR |= 0x00010000;

- ADC için yazdığımız fonksiyon aşağıdaki gibidir.

```
32@void ADC_Config(void)
33 {
34     RCC->APB2ENR |= 0x00000100;          //ADC1
35
36     ADC1->CR1 |= 1 << 8;                //SCAN
37     ADC1->CR1 |= 0 << 24;              //RES 12-bit
38     ADC1->CR2 |= 1 << 0;                //ADON
39     ADC1->CR2 |= 1 << 1;              //CONT
40     ADC1->CR2 |= 1 << 8;                //DMA
41     ADC1->CR2 |= 1 << 9;              //DDS
42     ADC1->CR2 |= 1 << 10;             //EOCS
43     ADC1->CR2 |= 1 << 30;             //SWSTART
44     ADC1->SMPR2 |= 0x00000003;        //SMP0 56 cycles
45     ADC1->SQR1 |= 0 << 20;            //1 conversion
46     ADC1->SQR3 |= 0 << 0;            //SQ1
47     ADC->CCR |= 0x00010000.           //ADCPRE 4
```

```

32 void ADC_Config(void)
33 {
34     RCC->APB2ENR |= 0x00000100;           //ADC1
35
36     ADC1->CR1 |= 1 << 8;                //SCAN
37     ADC1->CR1 |= 0 << 24;               //RES 12-bit
38     ADC1->CR2 |= 1 << 0;                //ADON
39     ADC1->CR2 |= 1 << 1;                //CONT
40     ADC1->CR2 |= 1 << 8;                //DMA
41     ADC1->CR2 |= 1 << 9;                //DDS
42     ADC1->CR2 |= 1 << 10;               //EOCS
43     ADC1->CR2 |= 1 << 30;               //SWSTART
44     ADC1->SMPR2 |= 0x00000003;          //SMP0 56 cycles
45     ADC1->SQR1 |= 0 << 20;              //1 conversion
46     ADC1->SQR3 |= 0 << 0;              //SQ1
47     ADC->CCR |= 0x00010000;             //ADCPRE 4
48 }

```

- DMA'nın clock hattı AHB'ye gidiyor.



### RCC AHB1 peripheral clock enable register (RCC\_AHB1ENR)

Address offset: 0x30

Reset value: 0x0010 0000

Access: no wait state, word, half-word and byte access.

|               | 31                      | 30          | 29                  | 28                 | 27                 | 26           | 25         | 24          | 23          | 22               | 21      | 20            | 19          | 18          | 17          | 16 |
|---------------|-------------------------|-------------|---------------------|--------------------|--------------------|--------------|------------|-------------|-------------|------------------|---------|---------------|-------------|-------------|-------------|----|
| Reser-<br>ved | OTGH<br>S<br>ULPIE<br>N | OTGH<br>SEN | ETHM<br>ACPTP<br>EN | ETHM<br>ACRXE<br>N | ETHM<br>ACTXE<br>N | ETHMA<br>CEN | Reserved   | DMA2E<br>N  | DMA1E<br>N  | CCMDAT<br>ARAMEN | Res.    | BKPSR<br>AMEN | Reserved    |             |             |    |
|               | rw                      | rw          | rw                  | rw                 | rw                 | rw           |            | rw          | rw          |                  |         | rw            |             |             |             |    |
| 15            | 14                      | 13          | 12                  | 11                 | 10                 | 9            | 8          | 7           | 6           | 5                | 4       | 3             | 2           | 1           | 0           |    |
|               | Reserved                |             | CRCE<br>N           | Reserved           |                    |              | GPIOE<br>N | GPIOH<br>EN | GPIOG<br>EN | GPIOF<br>N       | GPIOEEN | GPIOD<br>EN   | GPIOC<br>EN | GPIO<br>BEN | GPIO<br>AEN |    |
|               |                         |             | rw                  |                    |                    |              | rw         | rw          | rw          | rw               | rw      | rw            | rw          | rw          | rw          |    |

Bit 22 DMA2EN: DMA2 clock enable

Set and cleared by software.

0: DMA2 clock disabled

1: DMA2 clock enabled

- 22.pini 1 yapıyoruz.

RCC->AHB1ENR |= 0x00400000;

- ADC için DMA2 kullanıyoruz. Tabloya baktığımızda biz burada Stream 4'ü seçiyoruz.

## DMA1 request mapping

| Peripheral requests | Stream 0                | Stream 1                | Stream 2            | Stream 3                | Stream 4              | Stream 5    | Stream 6                | Stream 7            |
|---------------------|-------------------------|-------------------------|---------------------|-------------------------|-----------------------|-------------|-------------------------|---------------------|
| Channel 0           | SPI3_RX                 | -                       | SPI3_RX             | SPI2_RX                 | SPI2_TX               | SPI3_TX     | -                       | SPI3_TX             |
| Channel 1           | I2C1_RX                 | -                       | TIM7_UP             | -                       | TIM7_UP               | I2C1_RX     | I2C1_TX                 | I2C1_TX             |
| Channel 2           | TIM4_CH1                | -                       | I2S3_EXT_RX         | TIM4_CH2                | I2S2_EXT_TX           | I2S3_EXT_TX | TIM4_UP                 | TIM4_CH3            |
| Channel 3           | I2S3_EXT_RX             | TIM2_UP<br>TIM2_CH3     | I2C3_RX             | I2S2_EXT_RX             | I2C3_TX               | TIM2_CH1    | TIM2_CH2<br>TIM2_CH4    | TIM2_UP<br>TIM2_CH4 |
| Channel 4           | UART5_RX                | USART3_RX               | UART4_RX            | USART3_TX               | UART4_TX              | USART2_RX   | USART2_TX               | UART5_TX            |
| Channel 5           | UART8_TX <sup>(1)</sup> | UART7_TX <sup>(1)</sup> | TIM3_CH4<br>TIM3_UP | UART7_RX <sup>(1)</sup> | TIM3_CH1<br>TIM3_TRIG | TIM3_CH2    | UART8_RX <sup>(1)</sup> | TIM3_CH3            |
| Channel 6           | TIM5_CH3<br>TIM5_UP     | TIM5_CH4<br>TIM5_TRIG   | TIM5_CH1            | TIM5_CH4<br>TIM5_TRIG   | TIM5_CH2              | -           | TIM5_UP                 | -                   |
| Channel 7           | -                       | TIM6_UP                 | I2C2_RX             | I2C2_RX                 | USART3_TX             | DAC1        | DAC2                    | I2C2_TX             |

1. These requests are available on STM32F42xxx and STM32F43xxx only.

## DMA2 request mapping

| Peripheral requests | Stream 0               | Stream 1               | Stream 2                         | Stream 3               | Stream 4                          | Stream 5               | Stream 6                         | Stream 7                          |
|---------------------|------------------------|------------------------|----------------------------------|------------------------|-----------------------------------|------------------------|----------------------------------|-----------------------------------|
| Channel 0           | ADC1                   | SAI1_A <sup>(1)</sup>  | TIM8_CH1<br>TIM8_CH2<br>TIM8_CH3 | SAI1_A <sup>(1)</sup>  | ADC1                              | SAI1_B <sup>(1)</sup>  | TIM1_CH1<br>TIM1_CH2<br>TIM1_CH3 | -                                 |
| Channel 1           | -                      | DCMI                   | ADC2                             | ADC2                   | SAI1_B <sup>(1)</sup>             | SPI6_TX <sup>(1)</sup> | SPI6_RX <sup>(1)</sup>           | DCMI                              |
| Channel 2           | ADC3                   | ADC3                   | -                                | SPI5_RX <sup>(1)</sup> | SPI5_TX <sup>(1)</sup>            | CRYP_OUT               | CRYP_IN                          | HASH_IN                           |
| Channel 3           | SPI1_RX                | -                      | SPI1_RX                          | SPI1_TX                | -                                 | SPI1_TX                | -                                | -                                 |
| Channel 4           | SPI4_RX <sup>(1)</sup> | SPI4_TX <sup>(1)</sup> | USART1_RX                        | SDIO                   | -                                 | USART1_RX              | SDIO                             | USART1_TX                         |
| Channel 5           | -                      | USART6_RX              | USART6_RX                        | SPI4_RX <sup>(1)</sup> | SPI4_TX <sup>(1)</sup>            | -                      | USART6_TX                        | USART6_TX                         |
| Channel 6           | TIM1_TRIG              | TIM1_CH1               | TIM1_CH2                         | TIM1_CH1               | TIM1_CH4<br>TIM1_TRIG<br>TIM1_COM | TIM1_UP                | TIM1_CH3                         | -                                 |
| Channel 7           | -                      | TIM8_UP                | TIM8_CH1                         | TIM8_CH2               | TIM8_CH3                          | SPI5_RX <sup>(1)</sup> | SPI5_TX <sup>(1)</sup>           | TIM8_CH4<br>TIM8_TRIG<br>TIM8_COM |

1. These requests are available on STM32F42xxx and STM32F43xxx.

### DMA stream x configuration register (DMA\_SxCR) (x = 0..7)

This register is used to configure the concerned stream.

Address offset:  $0x10 + 0x18 \times \text{stream number}$

Reset value: 0x0000 0000

- 0.bit 0 olduğunda konfigürasyon yapmaya imkan tanıyor. Bunun için okuma yapacağız.

Bit 0 **EN**: Stream enable / flag stream ready when read low

This bit is set and cleared by software.

0: Stream disabled

1: Stream enabled

This bit may be cleared by hardware:

- on a DMA end of transfer (stream ready to be configured)
- if a transfer error occurs on the AHB master buses
- when the FIFO threshold on memory AHB port is not compatible with the size of the burst

When this bit is read as 0, the software is allowed to program the Configuration and FIFO bits registers. It is forbidden to write these registers when the EN bit is read as 1.

*Note: Before setting EN bit to '1' to start a new transfer, the event flags corresponding to the stream in DMA\_LISR or DMA\_HISR register must be cleared.*

- 1 olduğu sürece 0 olana kadar bekliyoruz. 0 olunca döngüden çıkış alt satıra geçecek.

```
while((DMA2_Stream4->CR & 0x00000001) == 1);
```

- ADC'den RAM'e yazacağımız bunun için Peripheral-to-memory seçiyoruz.

Bits 7:6 **DIR[1:0]**: Data transfer direction

This bit is set and cleared by software.

00: Peripheral-to-memory

01: Memory-to-peripheral

10: Memory-to-memory

11: reserved

These bits are protected and can be written only if EN is '0'.

```
DMA2_Stream4->CR |= 0 << 6;
```

- Sürekli transfer için 8.biti 1 yapıyoruz.

Bit 8 **CIRC**: Circular mode

This bit is set and cleared by software and can be cleared by hardware.

0: Circular mode disabled

1: Circular mode enabled

When the peripheral is the flow controller (bit PFCTRL=1) and the stream is enabled (bit EN=1), then this bit is automatically forced by hardware to 0.

It is automatically forced by hardware to 1 if the DBM bit is set, as soon as the stream is enabled (bit EN ='1').

```
DMA2_Stream4->CR |= 1 << 8;
```

- Adresin değişmesini istemiyoruz. 9.biti 0 yapıyoruz.

Bit 9 **PINC**: Peripheral increment mode

This bit is set and cleared by software.

0: Peripheral address pointer is fixed

1: Peripheral address pointer is incremented after each data transfer (increment is done according to PSIZE)

This bit is protected and can be written only if EN is '0'.

```
DMA2_Stream4->CR |= 0 << 9;
```

Bit 10 **MINC**: Memory increment mode

This bit is set and cleared by software.

0: Memory address pointer is fixed

1: Memory address pointer is incremented after each data transfer (increment is done according to MSIZE)

This bit is protected and can be written only if EN is '0'.

```
DMA2_Stream4->CR |= 1 << 10;
```

- Peripheral ve Memory için data boyutunu 32 bit belirliyoruz.

Bits 12:11 **PSIZE[1:0]**: Peripheral data size

These bits are set and cleared by software.

- 00: Byte (8-bit)
  - 01: Half-word (16-bit)
  - 10: Word (32-bit)
  - 11: reserved

These bits are protected and can be written only if EN is '0'

Bits 14:13 **MSIZE[1:0]**: Memory data size

These bits are set and cleared by software.

- 00: byte (8-bit)
  - 01: half-word (16-bit)
  - 10: word (32-bit)
  - 11: reserved

These bits are protected and can be written only if EN is '0'.

In direct mode, MSIZE is forced by hardware to the same value as PSIZE as soon as bit EN = '1'.

```
DMA2_Stream4->CR |= 2 << 11; //PSIZE Word (32-bit)  
DMA2_Stream4->CR |= 2 << 13; //MSIZE Word (32-bit)
```

- Önceliği çok yüksek yapıyoruz.

Bits 17:16 PI[1:0]: Priority level

These bits are set and cleared by software

- These bits are:  
00: Low  
01: Medium  
10: High  
11: Very high

These bits are protected and can be written only if EN is '0'.

DMA2\_Stream4->CB |= 3 << 16;

- Kanal secimi icin 0.kanal yapivoruz.

Bits 27:25 CHSEL[2:0]: Channel selection

These bits are set and cleared by software.

- ```
000: channel 0 selected  
001: channel 1 selected  
010: channel 2 selected  
011: channel 3 selected  
100: channel 4 selected  
101: channel 5 selected  
110: channel 6 selected  
111: channel 7 selected
```

These bits are protected and can be written only if EN is '0'

- Kaç tane çevresel birimden yani ADC'den okuma yaptığımızı belirtiyoruz.

DMA stream x number of data register (DMA\_SxNDTR) (x = 0..7)

Address offset:  $0x14 + 0x18 \times \text{stream number}$

Reset value: 0x0000 0000

#### Bits 15:0 **NDT[15:0]**: Number of data items to transfer

Number of data items to be transferred (0 up to 65535). This register can be written only when the stream is disabled. When the stream is enabled, this register is read-only, indicating the remaining data items to be transmitted. This register decrements after each DMA transfer.

Once the transfer has completed, this register can either stay at zero (when the stream is in normal mode) or be reloaded automatically with the previously programmed value in the following cases:

- when the stream is configured in Circular mode.
- when the stream is enabled again by setting EN bit to '1'

If the value of this register is zero, no transaction can be served even if the stream is enabled.

- Biz bir tane birimden yani ADC1'den okuma yapıyoruz.

`DMA2_Stream4->NDTR |= 1;`

- Okuma yaptığımız Çevresel birimin adresini yazıyoruz.

#### **DMA stream x peripheral address register (DMA\_SxPAR) (x = 0..7)**

Address offset:  $0x18 + 0x18 \times \text{stream number}$

Reset value: 0x0000 0000

| 31         | 30 | 29 | 28 | 27 | 26 | 25 | 24 | 23 | 22 | 21 | 20 | 19 | 18 | 17 | 16 |
|------------|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|
| PAR[31:16] |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| rw         | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw |
| 15         | 14 | 13 | 12 | 11 | 10 | 9  | 8  | 7  | 6  | 5  | 4  | 3  | 2  | 1  | 0  |
| PAR[15:0]  |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| rw         | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw |

#### Bits 31:0 **PAR[31:0]**: Peripheral address

Base address of the peripheral data register from/to which the data will be read/written.

These bits are write-protected and can be written only when bit EN = '0' in the DMA\_SxCR register.

DMA transfer.

Once the transfer has completed, this register can either stay at zero (when the stream is in normal mode) or be reloaded automatically with the previously programmed value in the following cases:

- when the stream is configured in Circular mode.
- when the stream is enabled again by setting EN bit to '1'

If the value of this register is zero, no transaction can be served even if the stream is enabled.

#### **ADC regular data register (ADC\_DR)**

Address offset: 0x4C

Reset value: 0x0000 0000

| 31         | 30 | 29 | 28 | 27 | 26 | 25 | 24 | 23 | 22 | 21 | 20 | 19 | 18 | 17 | 16 |
|------------|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|
| Reserved   |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| 15         | 14 | 13 | 12 | 11 | 10 | 9  | 8  | 7  | 6  | 5  | 4  | 3  | 2  | 1  | 0  |
| DATA[15:0] |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| r          | r  | r  | r  | r  | r  | r  | r  | r  | r  | r  | r  | r  | r  | r  | r  |

Bits 31:16 Reserved, must be kept at reset value.

#### Bits 15:0 **DATA[15:0]**: Regular data

These bits are read-only. They contain the conversion result from the regular channels. The data are left- or right-aligned as shown in [Figure 48](#) and [Figure 49](#).

`DMA2_Stream4->PAR |= (uint8_t) &ADC1->DR;`

- Değişkenin adresini yazıyoruz.

## DMA stream x memory 0 address register (DMA\_SxM0AR) ( $x = 0..7$ )

Address offset:  $0x1C + 0x18 \times \text{stream number}$

Reset value: 0x0000 0000

| 31         | 30 | 29 | 28 | 27 | 26 | 25 | 24 | 23 | 22 | 21 | 20 | 19 | 18 | 17 | 16 |
|------------|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|
| M0A[31:16] |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| rw         | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw |
| 15         | 14 | 13 | 12 | 11 | 10 | 9  | 8  | 7  | 6  | 5  | 4  | 3  | 2  | 1  | 0  |
| M0A[15:0]  |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| rw         | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw |

Bits 31:0 **M0A[31:0]**: Memory 0 address

Base address of Memory area 0 from/to which the data will be read/written.

These bits are write-protected. They can be written only if:

- the stream is disabled (bit EN= '0' in the DMA\_SxCR register) or
- the stream is enabled (EN='1' in DMA\_SxCR register) and bit CT = '1' in the DMA\_SxCR register (in Double buffer mode).

```

3 uint8_t adc;
4 uint8_t adc1[8];

DMA2_Stream4->M0AR |= (uint8_t) &adc1;

```

## DMA stream x FIFO control register (DMA\_SxFCR) ( $x = 0..7$ )

Address offset:  $0x24 + 0x24 \times \text{stream number}$

Reset value: 0x0000 0021

| 31       | 30 | 29 | 28 | 27 | 26 | 25 | 24 | 23        | 22 | 21      | 20 | 19 | 18    | 17 | 16       |  |
|----------|----|----|----|----|----|----|----|-----------|----|---------|----|----|-------|----|----------|--|
| Reserved |    |    |    |    |    |    |    |           |    |         |    |    |       |    |          |  |
| 15       | 14 | 13 | 12 | 11 | 10 | 9  | 8  | 7         | 6  | 5       | 4  | 3  | 2     | 1  | 0        |  |
| Reserved |    |    |    |    |    |    |    |           |    |         |    |    |       |    |          |  |
| FEIE     |    |    |    |    |    |    |    | Reser ved |    | FS[2:0] |    |    | DMDIS |    | FTH[1:0] |  |
|          |    |    |    |    |    |    |    | r         | r  | r       | r  | r  | rw    | rw | rw       |  |

Bits 1:0 **FTH[1:0]**: FIFO threshold selection

These bits are set and cleared by software.

- 00: 1/4 full FIFO
- 01: 1/2 full FIFO
- 10: 3/4 full FIFO
- 11: full FIFO

These bits are not used in the direct mode when the DMIS value is zero.

These bits are protected and can be written only if EN is '0'.

DMA2\_Stream4->FCR |= 0 << 1;

- Stream4'ü fonksiyonun en son satırında aktif ediyoruz.

DMA2\_Stream4->CR |= 1 << 0;

- Adres tanımları fonksiyonun ortalarında yazmak yerine while döngüsünün sonrasında yazdık.
- DMA için yazdığımız fonksiyon aşağıdaki gibidir.

```

53 void DMA_Config(void)
54 {
55     RCC->AHB1ENR |= 0x00400000;           //DMA2
56
57     while((DMA2_Stream4->CR & 0x00000001) == 1);
58     DMA2_Stream4->PAR |= (uint8_t) &ADC1->DR;          //PAR Peripheral address
59     DMA2_Stream4->M0AR |= (uint8_t) &adc1;             //M0A Memory 0 address
60     DMA2_Stream4->CR |= 0 << 6;                      //DIR
61     DMA2_Stream4->CR |= 1 << 8;                      //CIRC
62     DMA2_Stream4->CR |= 0 << 9;                      //PINC
63     DMA2_Stream4->CR |= 1 << 10;                     //PSIZE Word (32-bit)
64     DMA2_Stream4->CR |= 2 << 11;                     //MSIZE Word (32-bit)
65     DMA2_Stream4->CR |= 2 << 13;                     //PL
66     DMA2_Stream4->CR |= 3 << 16;                     //CHSEL
67     DMA2_Stream4->CR |= 0 << 25;                     //NDT
68     DMA2_Stream4->NDTR |= 1;                         //FTH 1/2 full FIFO
69     DMA2_Stream4->FCR |= 0 << 1;                      //EN Stream enabled
70     DMA2_Stream4->CR |= 1 << 0;
71 }

```

## Kod Kısmı

- ADC için çevrimi DMA'dan sonra başlatmak için 80.satırı tekrar yazdık.

```

73 int main(void)
74 {
75     RCC_Config();
76     GPIO_Config();
77     ADC_Config();
78     DMA_Config();
79     //ADC1->CR2 |= ADC_CR2_SWSTART;
80     ADC1->CR2 |= 0x40000000;
81
82     while (1)
83     {
84
85     }
86 }

```

# 06\_01 Timer Değer Okuma

25 Aralık 2021 Cumartesi 00:59

## 06\_01 Timer Değer Okuma

### ➤ REGISTER

#### Konfigürasyon Kısmı

- TIM2'nin clock hattı APB1'e gitiyor.



## RCC APB1 peripheral clock enable register (RCC\_APB1ENR)

Address offset: 0x40

Reset value: 0x0000 0000

Access: no wait state, word, half-word and byte access.

|          | 31      | 30       | 29      | 28            | 27       | 26       | 25            | 24      | 23      | 22      | 21       | 20       | 19         | 18         | 17            | 16 |
|----------|---------|----------|---------|---------------|----------|----------|---------------|---------|---------|---------|----------|----------|------------|------------|---------------|----|
| Reserved |         | DAC EN   | PWR EN  | Reser-<br>ved | CAN2 EN  | CAN1 EN  | Reser-<br>ved | I2C3 EN | I2C2 EN | I2C1 EN | UART5 EN | UART4 EN | USART 3 EN | USART 2 EN | Reser-<br>ved |    |
|          |         | rw       | rw      |               | rw       | rw       |               | rw      | rw      | rw      | rw       | rw       | rw         | rw         |               | rw |
|          | 15      | 14       | 13      | 12            | 11       | 10       | 9             | 8       | 7       | 6       | 5        | 4        | 3          | 2          | 1             | 0  |
| SPI3 EN  | SPI2 EN | Reserved | WWDG EN | Reserved      | TIM14 EN | TIM13 EN | TIM12 EN      | TIM7 EN | TIM6 EN | TIM5 EN | TIM4 EN  | TIM3 EN  | TIM2 EN    |            |               |    |
| rw       | rw      |          | rw      |               | rw       | rw       | rw            | rw      | rw      | rw      | rw       | rw       | rw         | rw         | rw            | rw |

0.biti 1 yapıyoruz.

Bit 0 **TIM2EN**: TIM2 clock enable

Set and cleared by software.

0: TIM2 clock disabled

1: TIM2 clock enabled

`RCC->APB1ENR |= 0x00000001;`

## TIMx control register 1 (TIMx\_CR1)

Address offset: 0x00

Reset value: 0x0000

|          | 15 | 14 | 13 | 12 | 11 | 10 | 9        | 8    | 7   | 6   | 5   | 4   | 3    | 2   | 1  | 0  |
|----------|----|----|----|----|----|----|----------|------|-----|-----|-----|-----|------|-----|----|----|
| Reserved |    |    |    |    |    |    | CKD[1:0] | ARPE | CMS | DIR | OPM | URS | UDIS | CEN |    |    |
|          |    |    |    |    |    |    | rw       | rw   | rw  | rw  | rw  | rw  | rw   | rw  | rw | rw |

- Sayma yapacağımdan saymayı başlatmak için 0.biti aktif ediyoruz. Sayma başlayacağından bu işlem fonksiyonun en son satırında olmalı.

Bit 0 **CEN**: Counter enable

0: Counter disabled

1: Counter enabled

*Note: External clock, gated mode and encoder mode can work only if the CEN bit has been previously set by software. However trigger mode can set the CEN bit automatically by hardware.*

CEN is cleared automatically in one-pulse mode, when an update event occurs.

`TIM2->CR1 |= 1 << 0;`

- Saymanın yönünü belirliyoruz. Biz yukarı doğru saymasını istiyoruz.

Bit 4 **DIR**: Direction

0: Counter used as upcounter

1: Counter used as downcounter

*Note: This bit is read only when the timer is configured in Center-aligned mode or Encoder mode.*

`TIM2->CR1 |= 0 << 4;`

- 5. ve 6.biti 0 yapıyoruz.

Bits 6:5 **CMS**: Center-aligned mode selection

00: Edge-aligned mode. The counter counts up or down depending on the direction bit (DIR).

01: Center-aligned mode 1. The counter counts up and down alternatively. Output compare interrupt flags of channels configured in output (CCxS=00 in TIMx\_CCMRx register) are set only when the counter is counting down.

10: Center-aligned mode 2. The counter counts up and down alternatively. Output compare interrupt flags of channels configured in output (CCxS=00 in TIMx\_CCMRx register) are set only when the counter is counting up.

11: Center-aligned mode 3. The counter counts up and down alternatively. Output compare interrupt flags of channels configured in output (CCxS=00 in TIMx\_CCMRx register) are set both when the counter is counting up or down.

*Note: It is not allowed to switch from edge-aligned mode to center-aligned mode as long as the counter is enabled (CEN=1)*

- Sistem clock 168MHz ayarlamıştık ve TIM2 clock veri yolu ise 42MHz'dir fakat bunun 2 katı değeri alıyorlardı yani 84MHz'dir. Biz bunu bölmek istiyorsak ayarlayabiliyoruz. Biz bölmüyoruz.

Bits 9:8 **CKD**: Clock division

This bit-field indicates the division ratio between the timer clock (CK\_INT) frequency and sampling clock used by the digital filters (ETR, TIx),

00:  $t_{DTS} = t_{CK\_INT}$

01:  $t_{DTS} = 2 \times t_{CK\_INT}$

10:  $t_{DTS} = 4 \times t_{CK\_INT}$

11: Reserved

- 8. ve 9.biti 0 yapıyoruz.

TIM2->**CR1** |= 0 << 8;

### **TIMx slave mode control register (TIMx\_SMCR)**

Address offset: 0x08

Reset value: 0x0000

| 15  | 14  | 13        | 12 | 11       | 10 | 9  | 8  | 7   | 6       | 5  | 4  | 3    | 2        | 1  | 0  |
|-----|-----|-----------|----|----------|----|----|----|-----|---------|----|----|------|----------|----|----|
| ETP | ECE | ETPS[1:0] |    | ETF[3:0] |    |    |    | MSM | TS[2:0] |    |    | Res. | SMS[2:0] |    |    |
| rw  | rw  | rw        | rw | rw       | rw | rw | rw | rw  | rw      | rw | rw |      | rw       | rw | rw |

- Slave mode çalışmayacağız.

#### Bits 2:0 **SMS**: Slave mode selection

When external signals are selected the active edge of the trigger signal (TRGI) is linked to the polarity selected on the external input (see Input Control register and Control Register description).

000: Slave mode disabled - if CEN = '1 then the prescaler is clocked directly by the internal clock.

001: Encoder mode 1 - Counter counts up/down on TI2FP1 edge depending on TI1FP2 level.

010: Encoder mode 2 - Counter counts up/down on TI1FP2 edge depending on TI2FP1 level.

011: Encoder mode 3 - Counter counts up/down on both TI1FP1 and TI2FP2 edges depending on the level of the other input.

100: Reset Mode - Rising edge of the selected trigger input (TRGI) reinitializes the counter and generates an update of the registers.

101: Gated Mode - The counter clock is enabled when the trigger input (TRGI) is high. The counter stops (but is not reset) as soon as the trigger becomes low. Both start and stop of the counter are controlled.

110: Trigger Mode - The counter starts at a rising edge of the trigger TRGI (but it is not reset). Only the start of the counter is controlled.

111: External Clock Mode 1 - Rising edges of the selected trigger (TRGI) clock the counter.

*Note: The gated mode must not be used if TI1F\_ED is selected as the trigger input (TS=100). Indeed, TI1F\_ED outputs 1 pulse for each transition on TI1F, whereas the gated mode checks the level of the trigger signal.*

*The clock of the slave timer must be enabled prior to receiving events from the master timer, and must not be changed on-the-fly while triggers are received from the master timer.*

```
TIM2->SMCR |= 0 << 0;
```

#### TIMx event generation register (TIMx\_EGR)

Address offset: 0x14

Reset value: 0x0000

| 15       | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5  | 4    | 3    | 2    | 1    | 0    |    |
|----------|----|----|----|----|----|---|---|---|---|----|------|------|------|------|------|----|
| Reserved |    |    |    |    |    |   |   |   |   | TG | Res. | CC4G | CC3G | CC2G | CC1G | UG |
| w        |    |    |    |    |    |   |   |   | w | w  | w    | w    | w    | w    | w    |    |

- Sayma dolduğunda sıfırlaması için 1 yaparız. 0.bit 1 yapıyoruz.

Bit 0 **UG**: Update generation

This bit can be set by software, it is automatically cleared by hardware.

0: No action

1: Re-initialize the counter and generates an update of the registers. Note that the prescaler counter is cleared too (anyway the prescaler ratio is not affected). The counter is cleared if the center-aligned mode is selected or if DIR=0 (upcounting), else it takes the auto-reload value (TIMx\_ARR) if DIR=1 (downcounting).

```
TIM2->EGR |= 1 << 0;
```

#### TIMx prescaler (TIMx\_PSC)

Address offset: 0x28

Reset value: 0x0000

| 15        | 14 | 13 | 12 | 11 | 10 | 9  | 8  | 7  | 6  | 5  | 4  | 3  | 2  | 1  | 0  |
|-----------|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|
| PSC[15:0] |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| rw        | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw |

- Prescaler bizim sayısının en üst seviyesini belirler.
- Sayma işlemi 0'dan başlamayıp 1'den başladığından 1 eksigini alarak yazarız.
- TIM2 clock hızı 84MHz olduğundan bunu kaç'a bölmeliyim diye soruyoruz. Bu clock hız için 42000'e bölüyoruz. Bu sayı da 41999 sayısı yapıyor.

Bits 15:0 **PSC[15:0]**: Prescaler value

The counter clock frequency CK\_CNT is equal to  $f_{CK\_PSC} / (PSC[15:0] + 1)$ .

PSC contains the value to be loaded in the active prescaler register at each update event (including when the counter is cleared through UG bit of TIMx\_EGR register or through trigger controller when configured in "reset mode").

TIM2->**PSC** |= 41999;

- 84MHz'i 42000'e böldüğümüzde 2000 sayısı yapıyor. Bu değer auto-reload oluyor. Bunun 1 eksigini yazıyoruz.

### **TIMx auto-reload register (TIMx\_ARR)**

Address offset: 0x2C

Reset value: 0xFFFF FFFF

| 31                               | 30 | 29 | 28 | 27 | 26 | 25 | 24 | 23 | 22 | 21 | 20 | 19 | 18 | 17 | 16 |
|----------------------------------|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|
| ARR[31:16] (depending on timers) |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| rw                               | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw |
| 15                               | 14 | 13 | 12 | 11 | 10 | 9  | 8  | 7  | 6  | 5  | 4  | 3  | 2  | 1  | 0  |
| ARR[15:0]                        |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| rw                               | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw |

Bits 31:16 **ARR[31:16]**: High auto-reload value (on TIM2 and TIM5).

Bits 15:0 **ARR[15:0]**: Auto-reload value

ARR is the value to be loaded in the actual auto-reload register.

Refer to the [Section 18.3.1: Time-base unit](#) for more details about ARR update and behavior.

The counter is blocked while the auto-reload value is null.

TIM2->**ARR** |= 1999;

- Böylece 1 sn'de 2000'e kadar sayıyor.

- RCC ve TIM için yazdığımız fonksiyonlar aşağıdaki gibidir.

```
5 void RCC_Config(void)
6 {
7     RCC->CR |= 0x00010000; //HSEON
8     while(!(RCC->CR & 0x00020000)); //HSERDY
9     RCC->CR |= 0x00080000; //CSSON
10    RCC->CFGR = 0x00000000;
11    RCC->PLLCFGR |= 0x00400000; //PLLSRC
12    RCC->PLLCFGR |= 0x00000004; //PLLM 4
13    RCC->PLLCFGR |= 0x00002A00; //PLLN 168
14    RCC->PLLCFGR |= 0x00000000; //PLLP 2
15    RCC->CR |= 0x01000000; //PLLON
16    while(!(RCC->CR & 0x02000000)); //PLLRDY
17    RCC->CFGR |= 0x00000001; //SW
18    while(!(RCC->CR & 0x00000001)); //SWS
19    RCC->CFGR |= 0x00000000; //HPRE AHB 1
20    RCC->CFGR |= 0x00001400; //PPRE1 APB1 4
21    RCC->CFGR |= 0x00008000; //PPRE2 APB2 2
22    RCC->CIR |= 0x00080000; //HSERDYC
23    RCC->CIR |= 0x00800000; //CSSC
24 }
```

```

26 void TIM_Config(void)
27 {
28     RCC->APB1ENR |= 0x00000001;           //TIM2EN
29
30     TIM2->CR1 |= 0 << 4;                //DIR Counter used as up counter
31     TIM2->CR1 |= 0 << 5;                //CMS Edge-aligned mode
32     TIM2->CR1 |= 0 << 8;                //tDTS = tCK_INT 84MHz
33     TIM2->SMCR |= 0 << 0;              //SMS Slave mode disabled
34     TIM2->EGR |= 1 << 0;                //UG Update generation
35     TIM2->PSC |= 41999;                 //PSC Prescaler value
36     TIM2->ARR |= 1999;                  //ARR Auto-reload value
37     TIM2->CR1 |= 1 << 0;                //CEN Counter enabled
38 }

```

## Kod Kısmı

### TIMx counter (TIMx\_CNT)

Address offset: 0x24

Reset value: 0x0000 0000

| 31                               | 30 | 29 | 28 | 27 | 26 | 25 | 24 | 23 | 22 | 21 | 20 | 19 | 18 | 17 | 16 |
|----------------------------------|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|
| CNT[31:16] (depending on timers) |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| rw                               | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw |
| 15                               | 14 | 13 | 12 | 11 | 10 | 9  | 8  | 7  | 6  | 5  | 4  | 3  | 2  | 1  | 0  |
| CNT[15:0]                        |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| rw                               | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw |

- CNT registeri sayma işlemini burada tutuyor.
- Değişken atayıp bu değişkene CNT registerine eşitliyoruz.

```
3 uint16_t count = 0;
```

```

40 int main(void)
41 {
42     RCC_Config();
43     TIM_Config();
44
45     while (1)
46     {
47         count = TIM2->CNT;
48     }
49 }
50 }

```

| Variable Name | Address/Expression | Read Value |
|---------------|--------------------|------------|
| count         | 0x2000002c         | 1838       |

## 06\_02 Timer Interrupt

25 Aralık 2021 Cumartesi 00:59

## 06\_02 Timer Interrupt



## Konfigürasyon Kısımları



| Pin N... | Signal on... | GPIO out... | GPIO mode    | GPIO Pull...  | Maximum ... | User Label | Modified                            |
|----------|--------------|-------------|--------------|---------------|-------------|------------|-------------------------------------|
| PB3      | n/a          | Low         | Output Pu... | No pull-up... | Very High   | LED1       | <input checked="" type="checkbox"/> |
| PB4      | n/a          | Low         | Output Pu... | No pull-up... | Very High   | LED2       | <input checked="" type="checkbox"/> |
| PB5      | n/a          | Low         | Output Pu... | No pull-up... | Very High   | LED3       | <input checked="" type="checkbox"/> |
| PB6      | n/a          | Low         | Output Pu... | No pull-up... | Very High   | LED4       | <input checked="" type="checkbox"/> |

- Timers kısmından TIM2 seçimi yapılır. Ardından Mod kısmından sadece Clock Source kısmını Internal Clock yaparız.

|                   |                |   |
|-------------------|----------------|---|
| Slave Mode        | Disable        | ▼ |
| Trigger Source    | Disable        | ▼ |
| Clock Source      | Internal Clock | ▼ |
| Channel1          | Disable        | ▼ |
| Channel2          | Disable        | ▼ |
| Channel3          | Disable        | ▼ |
| Channel4          | Disable        | ▼ |
| Combined Channels | Disable        | ▼ |

- Sistem clock 180MHz ayarlamıştık ve TIM2 clock veri yolu ise 45MHz'dir. fakat bunun 2 katı değeri alıyorlardı yani 90MHz'dır.



- Daha sonra Parameter Settings'den TIM2 1 saniye aralıklarla tekrarlı şekilde çalıştıracak değerler girilir. Sayma işlemi 0'dan başlamayıp 1'den başladığından 1 eksigini alarak yazarız.
- Prescalaler bizim sayısının en üst seviyesini belirler. Burası 16 bit olduğundan en fazla 65535 yazabilirim. TIM2 clock hızı 90MHz olduğundan bunu kaça bölmeliyim diye soruyoruz. Bu clock hızı için 45000'e bölüyoruz. Bu sayı da 44999 sayısı yapıyor.
- 90MHz'i 45000'e böldüğümüzde 2000 sayısı yapıyor. Bu değer Auto-reload oluyor. Bunun da 1 eksigini yazıyoruz. Counter Period kısmında her seferinde taşıma işlemi bittiğinden sonra tekrar bunu yükler. Yükleyeceği değeri yazarız.
- Sayma şeklini Up belirleyip yukarı doğru sayıyor.
- Auto-reload preload kısmında Enabled diyerek sayma bittiğinde başa dönmesini sağlarız.

- 2000 değeri yazmasaydık. Sonuç 2000 Hz olacağından sonucunda 0,0005s yani 0,5 ms olacaktır.
- 1 sn=1000 ms

#### Counter Settings

|                                       |             |
|---------------------------------------|-------------|
| Prescaler (PSC - 16 bits value)       | 45000-1     |
| Counter Mode                          | Up          |
| Counter Period (AutoReload Register - | .2000-1     |
| Internal Clock Division (CKD)         | No Division |
| auto-reload preload                   | Enable      |

$$UpdateEvent = \frac{Timer_{clock}}{(Prescaler + 1)(Period + 1)}$$

$$UpdateEvent = \frac{90.000.000}{(45000)(2000)} = 1 \text{ Hz} = \frac{1}{1} \text{ s} = 1 \text{ s}$$

- NVIC Settings kısmından TIM2 global interrupt Enabled yapılır. Bununla her güncellemede, sayıyı bitirmede bir interrupt oluşmasını sağlıyoruz.

| NVIC Interrupt Table  | Enabled                             | Preemption Priority | Sub Priority |
|-----------------------|-------------------------------------|---------------------|--------------|
| TIM2 global interrupt | <input checked="" type="checkbox"/> | 0                   | 0            |

#### Kod Kısımları

- hal\_tim.c dosyasından Timer'ı Interrupt ile başlatmamız gerekiyor.

```

453 /**
454  * @brief Starts the TIM Base generation in interrupt mode.
455  * @param htim TIM Base handle
456  * @retval HAL status
457 */
458 HAL_StatusTypeDef HAL_TIM_Base_Start_IT(TIM_HandleTypeDef *htim)

```

```

95  /* USER CODE BEGIN 2 */
96  HAL_TIM_Base_Start_IT(&htim2);
97  /* USER CODE END 2 */

```

- 96.satır çalışınca it.c dosyasına gider. Burada TIM için geçerli fonksiyonu 1sn'de bir çalıştırır.

```

202 /**
203  * @brief This function handles TIM2 global interrupt.
204 */
205 void TIM2_IRQHandler(void)
206 {
207  /* USER CODE BEGIN TIM2_IRQHandler_0 */
208
209  /* USER CODE END TIM2_IRQHandler_0 */
210  HAL_TIM_IRQHandler(&htim2);
211  /* USER CODE BEGIN TIM2_IRQHandler_1 */
212
213  /* USER CODE END TIM2_IRQHandler_1 */
214 }

```

- 210.satır ile bu fonksiyon dallanır ve dallanan fonksiyon içerisinde CallBack fonksiyonu vardır. Bu fonksiyon sayesinde int main içerisinde de kodlarımlı yazabilirim.

```
2038/** @defgroup TIM_Exported_Functions_Group9 TIM Callbacks functions
2039 * @brief   TIM Callbacks functions
2040 * @{
2041 */
2042 /* Callback in non blocking modes (Interrupt and DMA) *****/
2043 void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim);
• 1sn aralıklarla ledi yakıp söndürüyor.
57/* Private user code -----
58 /* USER CODE BEGIN 0 */
59void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
60{
61    HAL_GPIO_TogglePin(GPIOB, LED1_Pin | LED2_Pin | LED3_Pin | LED4_Pin);
62}
63 /* USER CODE END 0 */
```

# 07\_01 PWM Kullanımı

25 Aralık 2021 Cumartesi 00:59

## 07\_01 PWM Kullanımı

➤ HAL

Konfigürasyon Kısmı



| Pin Name | Signal on Pin | GPIO output | GPIO mode      | GPIO Pull-up    | Maximum output | User Label               | Modified |
|----------|---------------|-------------|----------------|-----------------|----------------|--------------------------|----------|
| PB6      | TIM4_CH1      | n/a         | Alternate F... | No pull-up a... | Low            | <input type="checkbox"/> |          |
| PB7      | TIM4_CH2      | n/a         | Alternate F... | No pull-up a... | Low            | <input type="checkbox"/> |          |
| PB8      | TIM4_CH3      | n/a         | Alternate F... | No pull-up a... | Low            | <input type="checkbox"/> |          |
| PB9      | TIM4_CH4      | n/a         | Alternate F... | No pull-up a... | Low            | <input type="checkbox"/> |          |

- Timers kısmından TIM4 seçimi yapılır. Ardından Mod kısmından kanal seçimi yapılır.

Slave Mode: Disable

Trigger Source: Disable

Internal Clock

Channel1: PWM Generation CH1

Channel2: PWM Generation CH2

Channel3: PWM Generation CH3

Channel4: Forced Output CH4

Combined Channels: Disable

XOR activation

- Period değerimize göre kanal çıkışlarına Pulse değeri yazacağız.
- Period kısmını Duty Cycle en fazla 100 olduğundan Period kısmına 100-1 olarak gireriz.

Yani Period 100 ve Pulse değeri 50 girersek aslında %50 Duty Cycle olur.

- 10kHz için işlem sonucunda Prescaler 90000 girilir.
- 1 kHz=1000Hz

$$UpdateEvent = \frac{90.000.000}{(Prescaler + 1)(100)} = 10000 \text{ Hz} = 10 \text{ kHz}$$
$$Prescaler + 1 = 90$$

### Counter Settings

|                                               |             |
|-----------------------------------------------|-------------|
| Prescaler (PSC - 16 bits value)               | 90-1        |
| Counter Mode                                  | Up          |
| Counter Period (AutoReload Register - 16 bit) | 100-1       |
| Internal Clock Division (CKD)                 | No Division |
| auto-reload preload                           | Disable     |

|                          |               |
|--------------------------|---------------|
| PWM Generation Channel 1 |               |
| Mode                     | PWM mode 1    |
| Pulse (16 bits value)    | 0             |
| Output compare preload   | Enable        |
| Fast Mode                | Disable       |
| CH Polarity              | High          |
| PWM Generation Channel 2 |               |
| Mode                     | PWM mode 1    |
| Pulse (16 bits value)    | 0             |
| Output compare preload   | Enable        |
| Fast Mode                | Disable       |
| CH Polarity              | High          |
| PWM Generation Channel 3 |               |
| Mode                     | PWM mode 1    |
| Pulse (16 bits value)    | 0             |
| Output compare preload   | Enable        |
| Fast Mode                | Disable       |
| CH Polarity              | High          |
| Forced Output Channel 4  |               |
| Mode                     | Forced Active |
| Pulse (16 bits value)    | 0             |
| Output compare preload   | Disable       |
| CH Polarity              | High          |

- Mode 1 ile Mode 2 arasındaki fark High ile Low durumların tersi olmasınaydı.
- Kanal %25 ise Mode durumunda %25'te High %75'te Low oluyordu.  
%75 olduğunda %25'te Low %75'te High oluyor.
- Fast Mode durumun Disable olduğu zaman saymaya yaparken üst limite kadar sayıktan sonra 0'a doğru azala azala inerken Enabled olduğu zaman 0'a doğru birden iner.

### Kod Kısmı

- hal\_tim.c dosyasından Timer'ı PWM ile başlatmamız gerekiyor.

```

1439 /**
1440  * @brief Starts the PWM signal generation.
1441  * @param htim TIM handle
1442  * @param Channel TIM Channels to be enabled
1443  *   This parameter can be one of the following values:
1444  *     @arg TIM_CHANNEL_1: TIM Channel 1 selected
1445  *     @arg TIM_CHANNEL_2: TIM Channel 2 selected
1446  *     @arg TIM_CHANNEL_3: TIM Channel 3 selected
1447  *     @arg TIM_CHANNEL_4: TIM Channel 4 selected
1448  * @retval HAL status
1449 */
150 HAL_StatusTypeDef HAL_TIM_PWM_Start(TIM_HandleTypeDef *htim, uint32_t Channel)

91 /* USER CODE BEGIN 2 */
92 HAL_TIM_PWM_Start(&htim4, TIM_CHANNEL_1);
93 HAL_TIM_PWM_Start(&htim4, TIM_CHANNEL_2);
94 HAL_TIM_PWM_Start(&htim4, TIM_CHANNEL_3);
95 HAL_TIM_PWM_Start(&htim4, TIM_CHANNEL_4);
96 /* USER CODE END 2 */

45 /* USER CODE BEGIN PV */
46 int i;
47 /* USER CODE END PV */

```

```

1375 /**
1376  * @brief Set the TIM Capture Compare Register value on runtime without calling another time ConfigChannel function.
1377  * @param __HANDLE__ TIM handle.
1378  * @param __CHANNEL__ TIM Channels to be configured.
1379  *   This parameter can be one of the following values:
1380  *     @arg TIM_CHANNEL_1: TIM Channel 1 selected
1381  *     @arg TIM_CHANNEL_2: TIM Channel 2 selected
1382  *     @arg TIM_CHANNEL_3: TIM Channel 3 selected
1383  *     @arg TIM_CHANNEL_4: TIM Channel 4 selected
1384  * @param __COMPARE__ specifies the Capture Compare register new value.
1385  * @retval None
1386 */
1387 #define __HAL_TIM_SET_COMPARE(__HANDLE__, __CHANNEL__, __COMPARE__) \
1388 (((__CHANNEL__) == TIM_CHANNEL_1) ? ((__HANDLE__)->Instance->CCR1 = (__COMPARE__)) :\
1389 ((__CHANNEL__) == TIM_CHANNEL_2) ? ((__HANDLE__)->Instance->CCR2 = (__COMPARE__)) :\
1390 ((__CHANNEL__) == TIM_CHANNEL_3) ? ((__HANDLE__)->Instance->CCR3 = (__COMPARE__)) :\
1391 ((__HANDLE__)->Instance->CCR4 = (__COMPARE__)))

```

- Compare kısmına Pulse değerini yazıyoruz.

```

98 /* Infinite loop */
99 /* USER CODE BEGIN WHILE */
100 while (1)
101 {
102 /* USER CODE END WHILE */
103
104 /* USER CODE BEGIN 3 */
105 for(i=0;i<=1999;i++)
106 {
107     __HAL_TIM_SET_COMPARE(&htim4,TIM_CHANNEL_1,i);
108     __HAL_TIM_SET_COMPARE(&htim4,TIM_CHANNEL_2,i);
109     __HAL_TIM_SET_COMPARE(&htim4,TIM_CHANNEL_3,i);
110     __HAL_TIM_SET_COMPARE(&htim4,TIM_CHANNEL_4,i);
111     HAL_Delay(100);
112 }
113
114 }
115 /* USER CODE END 3 */
116 }

```

## ➤ REGISTER

### Konfigürasyon Kısımları

- RCC için yazdığımız fonksiyonlar aşağıdaki gibidir.

```

3 void RCC_Config(void)
4 {
5     RCC->CR |= 0x00010000;           //HSEON
6     while(!(RCC->CR & 0x00020000)); //HSERDY
7     RCC->CR |= 0x00080000;          //CSSON
8     RCC->CFGR = 0x00000000;
9     RCC->PLLCFGR |= 0x00400000;    //PLLSRC
10    RCC->PLLCFGR |= 0x00000004;    //PLLM 4
11    RCC->PLLCFGR |= 0x00002A00;    //PLLN 168
12    RCC->PLLCFGR |= 0x00000000;    //PLLP 2
13    RCC->CR |= 0x01000000;         //PLLON
14    while(!(RCC->CR & 0x02000000)); //PLLRDY
15    RCC->CFGR |= 0x00000001;       //SW
16    while(!(RCC->CR & 0x00000001)); //SWS
17    RCC->CFGR |= 0x00000000;       //HPRE AHB 1
18    RCC->CFGR |= 0x00001400;       //PPRE1 APB1 4
19    RCC->CFGR |= 0x00008000;       //PPRE2 APB2 2
20    RCC->CIR |= 0x00080000;        //HSERDYC
21    RCC->CIR |= 0x00800000;        //CSSC
22 }

```

- GPIO için çıkışlarını alternatif fonksiyon yapıyoruz.

## GPIO port mode register (GPIOx\_MODER) (x = A..I/J/K)

Address offset: 0x00

Reset values:

- 0xA800 0000 for port A
- 0x0000 0280 for port B
- 0x0000 0000 for other ports

| 31           | 30 | 29           | 28 | 27           | 26 | 25           | 24 | 23           | 22 | 21           | 20 | 19          | 18 | 17          | 16 |
|--------------|----|--------------|----|--------------|----|--------------|----|--------------|----|--------------|----|-------------|----|-------------|----|
| MODER15[1:0] |    | MODER14[1:0] |    | MODER13[1:0] |    | MODER12[1:0] |    | MODER11[1:0] |    | MODER10[1:0] |    | MODER9[1:0] |    | MODER8[1:0] |    |
| rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw          | rw | rw          | rw |
| 15           | 14 | 13           | 12 | 11           | 10 | 9            | 8  | 7            | 6  | 5            | 4  | 3           | 2  | 1           | 0  |
| MODER7[1:0]  |    | MODER6[1:0]  |    | MODER5[1:0]  |    | MODER4[1:0]  |    | MODER3[1:0]  |    | MODER2[1:0]  |    | MODER1[1:0] |    | MODER0[1:0] |    |
| rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw          | rw | rw          | rw |

- Kart üzerindeki ledleri kullanıyoruz. 24., 26., 28. ve 30.biti 1 yapıyoruz.

Bits 2y:2y+1 **MODERy[1:0]**: Port x configuration bits (y = 0..15)

These bits are written by software to configure the I/O direction mode.

00: Input (reset state)

01: General purpose output mode

10: Alternate function mode

11: Analog mode

GPIOD->MODER |= 2 << 24 | 2 << 26 | 2 << 28 | 2 << 30;

- Alternatif fonksiyon ile kast edilen pinin çevresel birimlerden I2C, SPI olarak kullanılacağını belirtiyor. Biz burada TIM4 için kullanacağımızı belirteceğiz.

## GPIO alternate function low register (GPIOx\_AFRL) (x = A..I/J/K)

Address offset: 0x20

Reset value: 0x0000 0000

| 31         | 30 | 29 | 28 | 27         | 26 | 25 | 24 | 23         | 22 | 21 | 20 | 19         | 18 | 17 | 16 |
|------------|----|----|----|------------|----|----|----|------------|----|----|----|------------|----|----|----|
| AFRL7[3:0] |    |    |    | AFRL6[3:0] |    |    |    | AFRL5[3:0] |    |    |    | AFRL4[3:0] |    |    |    |
| rw         | rw | rw | rw | rw         | rw | rw | rw | rw         | rw | rw | rw | rw         | rw | rw | rw |
| 15         | 14 | 13 | 12 | 11         | 10 | 9  | 8  | 7          | 6  | 5  | 4  | 3          | 2  | 1  | 0  |
| AFRL3[3:0] |    |    |    | AFRL2[3:0] |    |    |    | AFRL1[3:0] |    |    |    | AFRL0[3:0] |    |    |    |
| rw         | rw | rw | rw | rw         | rw | rw | rw | rw         | rw | rw | rw | rw         | rw | rw | rw |

## GPIO alternate function high register (GPIOx\_AFRH)

(x = A..I/J)

Address offset: 0x24

Reset value: 0x0000 0000

| 31          | 30 | 29 | 28 | 27          | 26 | 25 | 24 | 23          | 22 | 21 | 20 | 19          | 18 | 17 | 16 |
|-------------|----|----|----|-------------|----|----|----|-------------|----|----|----|-------------|----|----|----|
| AFRH15[3:0] |    |    |    | AFRH14[3:0] |    |    |    | AFRH13[3:0] |    |    |    | AFRH12[3:0] |    |    |    |
| rw          | rw | rw | rw | rw          | rw | rw | rw | rw          | rw | rw | rw | rw          | rw | rw | rw |
| 15          | 14 | 13 | 12 | 11          | 10 | 9  | 8  | 7           | 6  | 5  | 4  | 3           | 2  | 1  | 0  |
| AFRH11[3:0] |    |    |    | AFRH10[3:0] |    |    |    | AFRH9[3:0]  |    |    |    | AFRH8[3:0]  |    |    |    |
| rw          | rw | rw | rw | rw          | rw | rw | rw | rw          | rw | rw | rw | rw          | rw | rw | rw |

- Low ile 0 ile 7.pinler arası iken high 8 ile 15.pinler arasıdır. Biz 12, 13,14 ve 15. pinleri kullandığımızdan high olanı kullanıyoruz.

Bits 31:0 **AFRHy**: Alternate function selection for port x bit y (y = 8..15)

These bits are written by software to configure alternate function I/Os

AFRHy selection:

|           |            |
|-----------|------------|
| 0000: AF0 | 1000: AF8  |
| 0001: AF1 | 1001: AF9  |
| 0010: AF2 | 1010: AF10 |
| 0011: AF3 | 1011: AF11 |
| 0100: AF4 | 1100: AF12 |
| 0101: AF5 | 1101: AF13 |
| 0110: AF6 | 1110: AF14 |
| 0111: AF7 | 1111: AF15 |

- Seçtiğimiz TIM4 çevresel birimi hangi AF'de olduğunu bilmek için Reference Manuel kitapçığına bakıyoruz.

### Selecting an alternate function

For pins 0 to 7, the GPIOx\_AFRL[31:0] register selects the dedicated alternate function



For pins 8 to 15, the GPIOx\_AFRH[31:0] register selects the dedicated alternate function



ai17538

- TIM4, AF2'de bulunuyor. Böylece pinlere AF2 için olan 0010 tanımlaması yapacağız.
- AFR'nin High ve Low olduğunda belirtmemiz gerekiyor. AFR'ye Ctrl ile sağ tıklarız. AFR'nin parantez içinde 2 elemanlı dizi olduğunu gösterir. 0.eleman low, 1.eleman high temsil eder. Biz 1 yazıyoruz.

682 `__IO uint32_t AFR[2]; /*!< GPIO alternate function registers, Address offset: 0x20-0x24 */`

`GPIOD->AFR[1] |= 2 << 16 | 2 << 20 | 2 << 24 | 2 << 28;`

- GPIO için yazdığımız fonksiyon aşağıdaki gibidir.

```
24 void GPIO_Config(void)
25 {
26     RCC->AHB1ENR |= 0x00000008; //D clock enable
27
28     GPIOD->MODER |= 2 << 24 | 2 << 26 | 2 << 28 | 2 << 30; //PD12, PD13, PD14, PD15
29     GPIOD->AFR[1] |= 2 << 16 | 2 << 20 | 2 << 24 | 2 << 28; //TIM4 AFRH12, AFRH13, AFRH14, AFRH15
30     GPIOD->OTYPER |= 0x00000000; //Output push-pull
31     GPIOD->OSPEEDR |= 0xFF000000; //Very high speed
32     GPIOD->PUPDR |= 0x00000000; //No pull-up, pull-down
33 }
```

- TIM4'nin clock hattı APB1'e gidiyor.





### RCC APB1 peripheral clock enable register (RCC\_APB1ENR)

Address offset: 0x40

Reset value: 0x0000 0000

Access: no wait state, word, half-word and byte access.

| 31       | 30      | 29       | 28        | 27       | 26       | 25        | 24       | 23       | 22      | 21       | 20       | 19         | 18         | 17        | 16 |
|----------|---------|----------|-----------|----------|----------|-----------|----------|----------|---------|----------|----------|------------|------------|-----------|----|
| Reserved | DAC EN  | PWR EN   | Reser-ved | CAN2 EN  | CAN1 EN  | Reser-ved | I2C3 EN  | I2C2 EN  | I2C1 EN | UART5 EN | UART4 EN | USART 3 EN | USART 2 EN | Reser-ved |    |
|          | rw      | rw       |           | rw       | rw       |           | rw       | rw       | rw      | rw       | rw       | rw         | rw         | rw        |    |
| 15       | 14      | 13       | 12        | 11       | 10       | 9         | 8        | 7        | 6       | 5        | 4        | 3          | 2          | 1         | 0  |
| SPI3 EN  | SPI2 EN | Reserved | WWDG EN   | Reserved | Reserved | TIM14 EN  | TIM13 EN | TIM12 EN | TIM7 EN | TIM6 EN  | TIM5 EN  | TIM4 EN    | TIM3 EN    | TIM2 EN   |    |
| rw       | rw      |          | rw        |          |          | rw        | rw       | rw       | rw      | rw       | rw       | rw         | rw         | rw        |    |

- 2.biti 1 yapıyoruz.

Bit 2 **TIM4EN**: TIM4 clock enable

Set and cleared by software.

0: TIM4 clock disabled

1: TIM4 clock enabled

RCC->APB1ENR |= 0x00000004; //TIM4EN

### TIMx control register 1 (TIMx\_CR1)

Address offset: 0x00

Reset value: 0x0000

| 15       | 14 | 13 | 12 | 11 | 10 | 9        | 8 | 7    | 6   | 5 | 4   | 3   | 2   | 1    | 0   |
|----------|----|----|----|----|----|----------|---|------|-----|---|-----|-----|-----|------|-----|
| Reserved |    |    |    |    |    | CKD[1:0] |   | ARPE | CMS |   | DIR | OPM | URS | UDIS | CEN |
|          |    |    |    |    |    |          |   | rw   |     |   | rw  | rw  | rw  | rw   | rw  |

- Sayma yapacağımızdan saymayı başlatmak için 0.biti aktif ediyoruz. Sayma başlayacağından bu işlem fonksiyonun en son satırında olmalı.

Bit 0 **CEN**: Counter enable

- 0: Counter disabled
- 1: Counter enabled

*Note: External clock, gated mode and encoder mode can work only if the CEN bit has been previously set by software. However trigger mode can set the CEN bit automatically by hardware.*

CEN is cleared automatically in one-pulse mode, when an update event occurs.

`TIM4->CR1 |= 1 << 0;`

- Saymanın yönünü belirliyoruz. Biz yukarı doğru saymasını istiyoruz.

Bit 4 **DIR**: Direction

- 0: Counter used as upcounter
- 1: Counter used as downcounter

*Note: This bit is read only when the timer is configured in Center-aligned mode or Encoder mode.*

`TIM4->CR1 |= 0 << 4;`

- 5. ve 6.biti 0 yapıyoruz.

Bits 6:5 **CMS**: Center-aligned mode selection

- 00: Edge-aligned mode. The counter counts up or down depending on the direction bit (DIR).
- 01: Center-aligned mode 1. The counter counts up and down alternatively. Output compare interrupt flags of channels configured in output (CCxS=00 in TIMx\_CCMRx register) are set only when the counter is counting down.
- 10: Center-aligned mode 2. The counter counts up and down alternatively. Output compare interrupt flags of channels configured in output (CCxS=00 in TIMx\_CCMRx register) are set only when the counter is counting up.
- 11: Center-aligned mode 3. The counter counts up and down alternatively. Output compare interrupt flags of channels configured in output (CCxS=00 in TIMx\_CCMRx register) are set both when the counter is counting up or down.

*Note: It is not allowed to switch from edge-aligned mode to center-aligned mode as long as the counter is enabled (CEN=1)*

`TIM4->CR1 |= 0 << 5;`

- Sistem clock 168MHz ayarlamıştık ve TIM4 clock veri yolu ise 42MHz'dir fakat bunun 2 katı değeri alıyorlardı yani 84MHz'dir. Biz bunu bölmek istiyorsak ayarlayabiliyoruz. Biz bölmüyoruz.

Bits 9:8 **CKD**: Clock division

This bit-field indicates the division ratio between the timer clock (CK\_INT) frequency and sampling clock used by the digital filters (ETR, TIx),

- 00:  $t_{DTs} = t_{CK\_INT}$
- 01:  $t_{DTs} = 2 \times t_{CK\_INT}$
- 10:  $t_{DTs} = 4 \times t_{CK\_INT}$
- 11: Reserved

- 8. ve 9.biti 0 yapıyoruz.

`TIM4->CR1 |= 0 << 8;`

- PWM aslında capture/compare mode kısmına giriyor.

### **TIMx capture/compare mode register 1 (TIMx\_CCMR1)**

Address offset: 0x18

Reset value: 0x0000

The channels can be used in input (capture mode) or in output (compare mode). The direction of a channel is defined by configuring the corresponding CCxS bits. All the other bits of this register have a different function in input and in output mode. For a given bit, OCxx describes its function when the channel is configured in output, ICxx describes its function when the channel is configured in input. Take care that the same bit can have a different meaning for the input stage and for the output stage.

| 15    | 14        | 13 | 12    | 11          | 10        | 9  | 8         | 7         | 6  | 5           | 4     | 3         | 2  | 1  | 0  |
|-------|-----------|----|-------|-------------|-----------|----|-----------|-----------|----|-------------|-------|-----------|----|----|----|
| OC2CE | OC2M[2:0] |    | OC2PE | OC2FE       | CC2S[1:0] |    | OC1CE     | OC1M[2:0] |    | OC1PE       | OC1FE | CC1S[1:0] |    |    |    |
|       | IC2F[3:0] |    |       | IC2PSC[1:0] |           |    | IC1F[3:0] |           |    | IC1PSC[1:0] |       |           |    |    |    |
| rw    | rw        | rw | rw    | rw          | rw        | rw | rw        | rw        | rw | rw          | rw    | rw        | rw | rw | rw |

- 1.kanal için output yapıyoruz yani ilk 2 bit 0 yapıyoruz.

Bits 1:0 **CC1S**: Capture/Compare 1 selection

This bit-field defines the direction of the channel (input/output) as well as the used input.

- 00: CC1 channel is configured as output.
- 01: CC1 channel is configured as input, IC1 is mapped on TI1.
- 10: CC1 channel is configured as input, IC1 is mapped on TI2.
- 11: CC1 channel is configured as input, IC1 is mapped on TRC. This mode is working only if an internal trigger input is selected through TS bit (TIMx\_SMCR register)

Bits 1:0 **CC1S**: Capture/Compare 1 selection

This bit-field defines the direction of the channel (input/output) as well as the used input.

00: CC1 channel is configured as output.

01: CC1 channel is configured as input, IC1 is mapped on TI1.

10: CC1 channel is configured as input, IC1 is mapped on TI2.

11: CC1 channel is configured as input, IC1 is mapped on TRC. This mode is working only if an internal trigger input is selected through TS bit (TIMx\_SMCR register)

Note: CC1S bits are writable only when the channel is OFF (CC1E = 0 in TIMx\_CCER).

- Mod olarak PWM mode 1 seçiyoruz. Bunun için bite 110 yazıyoruz.

Bits 6:4 **OC1M**: Output compare 1 mode

These bits define the behavior of the output reference signal OC1REF from which OC1 and OC1N are derived. OC1REF is active high whereas OC1 and OC1N active level depends on CC1P and CC1NP bits.

000: Frozen - The comparison between the output compare register TIMx\_CCR1 and the counter TIMx\_CNT has no effect on the outputs.(this mode is used to generate a timing base).

001: Set channel 1 to active level on match. OC1REF signal is forced high when the counter TIMx\_CNT matches the capture/compare register 1 (TIMx\_CCR1).

010: Set channel 1 to inactive level on match. OC1REF signal is forced low when the counter TIMx\_CNT matches the capture/compare register 1 (TIMx\_CCR1).

011: Toggle - OC1REF toggles when TIMx\_CNT=TIMx\_CCR1.

100: Force inactive level - OC1REF is forced low.

101: Force active level - OC1REF is forced high.

110: PWM mode 1 - In upcounting, channel 1 is active as long as TIMx\_CNT<TIMx\_CCR1 else inactive. In downcounting, channel 1 is inactive (OC1REF=0) as long as TIMx\_CNT>TIMx\_CCR1 else active (OC1REF=1).

111: PWM mode 2 - In upcounting, channel 1 is inactive as long as TIMx\_CNT<TIMx\_CCR1 else active. In downcounting, channel 1 is active as long as TIMx\_CNT>TIMx\_CCR1 else inactive.

Note: In PWM mode 1 or 2, the OCREF level changes only when the result of the comparison changes or when the output compare mode switches from "frozen" mode to "PWM" mode.

- 2.kanal için output yapıyoruz. 8. ve 9.biti 0 yapıyoruz.

Bits 9:8 **CC2S[1:0]**: Capture/Compare 2 selection

This bit-field defines the direction of the channel (input/output) as well as the used input.

00: CC2 channel is configured as output

01: CC2 channel is configured as input, IC2 is mapped on TI2

10: CC2 channel is configured as input, IC2 is mapped on TI1

11: CC2 channel is configured as input, IC2 is mapped on TRC. This mode is working only if an internal trigger input is selected through the TS bit (TIMx\_SMCR register)

Note: CC2S bits are writable only when the channel is OFF (CC2E = 0 in TIMx\_CCER).

- 2.kana'lın modunu 1.kanal'da yaptığımız gibi PWM Mode 1 yapıyoruz.

Bits 14:12 **OC2M[2:0]**: Output compare 2 mode

```
TIM4->CCMR1 |= 0 << 0 | 6 << 4 | 0 << 8 | 6 << 12;
```

### TIMx capture/compare mode register 2 (TIMx\_CCMR2)

Address offset: 0x1C

Reset value: 0x0000

Refer to the above CCMR1 register description.

| 15    | 14        | 13 | 12    | 11          | 10        | 9  | 8     | 7         | 6  | 5     | 4           | 3         | 2  | 1  | 0  |
|-------|-----------|----|-------|-------------|-----------|----|-------|-----------|----|-------|-------------|-----------|----|----|----|
| OC4CE | OC4M[2:0] |    | OC4PE | OC4FE       | CC4S[1:0] |    | OC3CE | OC3M[2:0] |    | OC3PE | OC3FE       | CC3S[1:0] |    |    |    |
|       | IC4F[3:0] |    |       | IC4PSC[1:0] |           |    |       | IC3F[3:0] |    |       | IC3PSC[1:0] |           |    |    |    |
| rw    | rw        | rw | rw    | rw          | rw        | rw | rw    | rw        | rw | rw    | rw          | rw        | rw | rw | rw |

- CCMR2 ile bu sefer kanal 3 ve 4 için yapıyoruz. Kanal 1 ve 2 ile yaptıklarımızın aynısını yapıyoruz.

Bits 1:0 **CC3S**: Capture/Compare 3 selection

This bit-field defines the direction of the channel (input/output) as well as the used input.

00: CC3 channel is configured as output

01: CC3 channel is configured as input, IC3 is mapped on TI3

10: CC3 channel is configured as input, IC3 is mapped on TI4

11: CC3 channel is configured as input, IC3 is mapped on TRC. This mode is working only if an internal trigger input is selected through TS bit (TIMx\_SMCR register)

Note: CC3S bits are writable only when the channel is OFF (CC3E = 0 in TIMx\_CCER).

Bits 6:4 **OC3M**: Output compare 3 mode

#### Bits 9:8 **CC4S**: Capture/Compare 4 selection

This bit-field defines the direction of the channel (input/output) as well as the used input.

00: CC4 channel is configured as output

01: CC4 channel is configured as input, IC4 is mapped on TI4

10: CC4 channel is configured as input, IC4 is mapped on TI3

11: CC4 channel is configured as input, IC4 is mapped on TRC. This mode is working only if an internal trigger input is selected through TS bit (TIMx\_SMCR register)

Note: CC4S bits are writable only when the channel is OFF (CC4E = 0 in TIMx\_CCER).

#### Bits 14:12 **OC4M**: Output compare 4 mode

`TIM4->CCMR2 |= 0 << 0 | 6 << 4 | 0 << 8 | 6 << 12;`

### TIMx capture/compare enable register (TIMx\_CCER)

Address offset: 0x20

Reset value: 0x0000

|             |            |            |            |             |            |            |            |             |            |            |            |             |            |            |            |
|-------------|------------|------------|------------|-------------|------------|------------|------------|-------------|------------|------------|------------|-------------|------------|------------|------------|
| 15          | 14         | 13         | 12         | 11          | 10         | 9          | 8          | 7           | 6          | 5          | 4          | 3           | 2          | 1          | 0          |
| CC4NP<br>rw | Res.<br>rw | CC4P<br>rw | CC4E<br>rw | CC3NP<br>rw | Res.<br>rw | CC3P<br>rw | CC3E<br>rw | CC2NP<br>rw | Res.<br>rw | CC2P<br>rw | CC2E<br>rw | CC1NP<br>rw | Res.<br>rw | CC1P<br>rw | CC1E<br>rw |

- Çıkışları Enabled yapıyoruz.

0., 4., 8. ve 12. bitleri 1 yapıyoruz.

Bit 0 **CC1E**: Capture/Compare 1 output enable.

CC1 channel configured as output:

0: Off - OC1 is not active

1: On - OC1 signal is output on the corresponding output pin

CC1 channel configured as input:

This bit determines if a capture of the counter value can actually be done into the input capture/compare register 1 (TIMx\_CCR1) or not.

0: Capture disabled

1: Capture enabled

Bit 4 **CC2E**: Capture/Compare 2 output enable.

refer to CC1E description

Bit 8 **CC3E**: Capture/Compare 3 output enable.

refer to CC1E description

Bit 12 **CC4E**: Capture/Compare 4 output enable.

refer to CC1E description

`TIM4->CCER |= 1 << 0 | 1 << 4 | 1 << 8 | 1 << 12;`

### TIMx prescaler (TIMx\_PSC)

Address offset: 0x28

Reset value: 0x0000

|           |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
|-----------|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|
| 15        | 14 | 13 | 12 | 11 | 10 | 9  | 8  | 7  | 6  | 5  | 4  | 3  | 2  | 1  | 0  |
| PSC[15:0] |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| rw        | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw |

- Prescalaaer bizim sayısının en üst seviyesini belirler.
- Sayma işlemi 0'dan başlamayıp 1'den başladığından 1 eksigini alarak yazarız.
- TIM4 clock hızı 84MHz olduğundan bunu kaça bölmeliyim diye soruyoruz. Bu clock hız için 42000'e böölüyoruz. Bu sayı da 41999 sayısı yapıyor.

Bits 15:0 **PSC[15:0]**: Prescaler value

The counter clock frequency CK\_CNT is equal to  $f_{CK\_PSC} / (PSC[15:0] + 1)$ .

PSC contains the value to be loaded in the active prescaler register at each update event (including when the counter is cleared through UG bit of TIMx\_EGR register or through trigger controller when configured in "reset mode").

`TIM4->PSC |= 41999;`

- 84MHz'i 42000'e böldüğümüzde 2000 sayısı yapıyor. Bu değer auto-reload oluyor. Bunun 1 eksigini yazıyoruz.

## **TIMx auto-reload register (TIMx\_ARR)**

Address offset: 0x2C

Reset value: 0xFFFF FFFF

Bits 31:16 **ARR[31:16]**: High auto-reload value (on TIM2 and TIM5).

Bits 15:0 **ARR[15:0]**: Auto-reload value

ARR is the value to be loaded in the actual auto-reload register.

Refer to the [Section 18.3.1: Time-base unit](#) for more details about ARR update and behavior.

The counter is blocked while the auto-reload value is null.

TIM4->ARR |= 1999;

- Böylece 1 sn'de 2000'e kadar sayıyor.
  - Pinlere atayacağımız paslar CCR1 12.pin, CCR2 13.pin, CCR3 14.pin ve CCR4 15.pin içindir.

## TIMx capture/compare register 1 (TIMx\_CCR1)

Address offset: 0x34

Reset value: 0x0000 0000

## TIMx capture/compare register 2 (TIMx\_CCR2)

Address offset: 0x38

Reset value: 0x0000 0000

## TIMx capture/compare register 3 (TIMx\_CCR3)

Address offset: 0x3C

Reset value: 0x0000 0000

## TIMx capture/compare register 4 (TIMx\_CCR4)

Address offset: 0x40

Reset value: 0x0000 0000

- Bunlara atayacağımız değerler en fazla Auto-reload değeri kadar olabilir.

```

35 void TIM_Config(void)
36 {
37     RCC->APB1ENR |= 0x00000004;                                //TIM4EN
38
39     TIM4->CR1 |= 0 << 4;                                     //DIR Counter used as up counter
40     TIM4->CR1 |= 0 << 5;                                     //CMS Edge-aligned mode
41     TIM4->CR1 |= 0 << 8;                                     //tDTS = tCK_INT 84MHz
42     TIM4->CCMR1 |= 0 << 0 | 6 << 4 | 0 << 8 | 6 << 12; //Capture/Compare selected output, PWM mode 1 (1 & 2)
43     TIM4->CCMR2 |= 0 << 0 | 6 << 4 | 0 << 8 | 6 << 12; //output, PWM mode 1 selected (3 & 4)
44     TIM4->CCER |= 1 << 0 | 1 << 4 | 1 << 8 | 1 << 12; //output enable (1, 2, 3 & 4 )
45     TIM4->PSC |= 41999;                                       //PSC Prescaler value
46     TIM4->ARR |= 1999;                                         //ARR Auto-reload value
47     TIM4->CCR1 |= 500;                                         //PD12
48     TIM4->CCR2 |= 1000;                                        //PD13
49     TIM4->CCR3 |= 1500;                                        //PD14
50     TIM4->CCR4 |= 1999;                                       //PD15
51     TIM4->CR1 |= 1 << 0;                                      //CEN Counter enabled
52 }
```

### Kod Kısımları

```

54 int main(void)
55 {
56     RCC_Config();
57     GPIO_Config();
58     TIM_Config();
59
60     while (1)
61     {
62
63     }
64 }
```

- Pindeki ledlerin durumu 12.pin %25, 13.pin %50, 14.pin %75, 15.pin %100 parlaklıktır yanıyor.

# 08\_01 USART ile Mesaj Gönderme

25 Aralık 2021 Cumartesi 01:00

## 08\_01 USART ile Mesaj Gönderme

### ➤ HAL

#### Teori

- Printf komutu ile veri göndermek için <https://www.youtube.com/watch?v=SpTh30wTmcM> ile <https://www.youtube.com/watch?v=WnCpPf7u4Xo> linklerindeki videoları inceleyebiliriz.
- Printf fonksiyonu ve türevleri hakkında bilgi almak için <https://bilgisayarkavramlari.com/2012/05/31/printf-sprintf-fprintf/> link üzerinden bakabiliriz.
- Nucleo kartlarının USART2 pini ile bağlantı yapıldığında, terminalden haberleşmeyi kartın USB portu ile sağlarız. USART2 dışındaki diğer USART pinlerini kullanırsak bu pinlerin çıkışına TTL dönüştürücü kullanmamız gereklidir.

#### Konfigürasyon Kısımları



| Pin N... | Signal on Pin | GPIO out... | GPIO mo...    | GPIO Pull...  | Maximum... | User L...                | Modified |
|----------|---------------|-------------|---------------|---------------|------------|--------------------------|----------|
| PA2      | USART2_TX     | n/a         | Alternate ... | No pull-up... | Very High  | <input type="checkbox"/> |          |
| PA3      | USART2_RX     | n/a         | Alternate ... | No pull-up... | Very High  | <input type="checkbox"/> |          |

Mode Asynchronous

- Sadece Transmit yapacağımızdan Data Direction kısmından Transmit Only seçeneğini seçebiliriz.

#### Basic Parameters

|             |                           |
|-------------|---------------------------|
| Baud Rate   | 9600 Bits/s               |
| Word Length | 8 Bits (including Parity) |
| Parity      | None                      |
| Stop Bits   | 1                         |

#### Advanced Parameters

|                |                      |
|----------------|----------------------|
| Data Direction | Receive and Transmit |
| Over Sampling  | 16 Samples           |

#### Kod Kısımları

- Sprintf komutu için stdio.h, strlen kütüphanesi için string.h kütüphanesini ekledik.

```
20 /* Includes -----  
21 #include "main.h"  
22 #include "stdio.h"  
23 #include "string.h"  
24 /* Private includes
```

- Göndereceğimiz ifadeleri tutacağımız char değişkenli dizi ekliyoruz.

```
46 /* USER CODE BEGIN PV */  
47 char tx_buff[50];  
48 uint8_t vize=65,final=87;  
49 float ortalama;  
50 /* USER CODE END PV */
```

- Float değeri yazdırma için bir ayar yapılmalıdır. Bunun için proje dosyasına sağ tıklayıp Properties tıklanır. C/C++ Build kısmından Settings kısmına gelinir ve buradan Tool Settings kısmından MCU Settings kısmından -

## u\_printf\_float kutucuğu işaretlenir.



- hal\_usart.c uzantılı dosyaya baktığımızda USART işlemini polling mode olarak kullanacağız.

```
1123 /**
1124 * @brief Sends an amount of data in blocking mode.
1125 * @note When UART parity is not enabled (PCE = 0), and Word Length is configured to 9 bits (M1-M0 = 01),
1126 *       the sent data is handled as a set of u16. In this case, Size must indicate the number
1127 *       of u16 provided through pData.
1128 * @param huart Pointer to a UART_HandleTypeDef structure that contains
1129 *               the configuration information for the specified UART module.
1130 * @param pData Pointer to data buffer (u8 or u16 data elements).
1131 * @param Size Amount of data elements (u8 or u16) to be sent
1132 * @param Timeout Timeout duration
1133 * @retval HAL status
1134 */
1135=HAL_StatusTypeDef HAL_UART_Transmit(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size, uint32_t Timeout)
```

- uint8\_t tipinde bir data göndermemizi istiyor fakat biz char değişkeni kullandığımızdan bir çevrim yapmamız gerekiyor.
- Sprintf komutu ile yazdırma yaparken fonksiyon içinde de kullanabiliriz ya da ayrı satırda yazıp daha sonra fonksiyon içinde boyutunu belirtmek için strlen fonksiyonunu kullanırız.

```
102 /* USER CODE BEGIN WHILE */
103 while (1)
104 {
105     /* USER CODE END WHILE */
106
107     /* USER CODE BEGIN 3 */
108     ortalama=(vize * 0.4) + (final * 0.6);
109     sprintf(tx_buff,"Vize: %d, Final:%d, Ortalama:%.2f\r\n",vize,final,ortalama);
110     HAL_UART_Transmit(&huart2, (uint8_t *)tx_buff, strlen(tx_buff), 100);
111     HAL_Delay(500);
112 }
113 /* USER CODE END 3 */
114 }
```

- Printf komutu ile yazdırmak istersek aşağıdaki kodu ekleriz. Bu kod ile while döngüsünde Transmit fonksiyonunu yazmamıza gerek kalmıyor.

```
60 /* USER CODE BEGIN 0 */
61 int _write(int file, char *ptr, int len)
62 {
63     HAL_UART_Transmit(&huart2, (uint8_t *)ptr, len, HAL_MAX_DELAY);
64     return len;
65 }
66 /* USER CODE END 0 */
```

- Ardından while döngüsünde yazdırma fonksiyonuna göre düzenleme yapıyoruz.

```

102 /* USER CODE BEGIN WHILE */
103 while (1)
104 {
105     /* USER CODE END WHILE */
106
107     /* USER CODE BEGIN 3 */
108     ortalama=(vize * 0.4) + (final * 0.6);
109     //sprintf(tx_buff,"Vize: %d, Final:%d, Ortalama:%.2f\r\n",vize,final,ortalama);
110     //HAL_UART_Transmit(&huart2, (uint8_t *)tx_buff, strlen(tx_buff), 100);
111     printf("Vize: %d, Final:%d, Ortalama:%.2f\r\n",vize,final,ortalama);
112     HAL_Delay(500);
113 }
114 /* USER CODE END 3 */
115 }
```



## ➤ REGISTER

### Konfigürasyon Kısmı

- RCC için yazdığımız fonksiyon aşağıdaki gibidir.

```

3 void RCC_Config(void)
4 {
5     RCC->CR |= 0x00010000;           //HSEON
6     while(!(RCC->CR & 0x00020000)); //HSERDY
7     RCC->CR |= 0x00080000;          //CSSON
8     RCC->CFGR = 0x00000000;
9     RCC->PLLCFGR |= 0x00400000;    //PLLSRC
10    RCC->PLLCFGR |= 0x00000004;    //PLLM 4
11    RCC->PLLCFGR |= 0x00002A00;    //PLLN 168
12    RCC->PLLCFGR |= 0x00000000;    //PLLP 2
13    RCC->CR |= 0x01000000;         //PLLON
14    while(!(RCC->CR & 0x02000000)); //PLLRDY
15    RCC->CFGR |= 0x00000001;       //SW
16    while(!(RCC->CR & 0x00000001)); //SWS
17    RCC->CFGR |= 0x00000000;       //HPRE AHB 1
18    RCC->CFGR |= 0x00001400;       //PPRE1 APB1 4
19    RCC->CFGR |= 0x00008000;       //PPRE2 APB2 2
20    RCC->CIR |= 0x00080000;        //HSERDYC
21    RCC->CIR |= 0x00800000;        //CSSC
22 }
```

- USART3 kullanacağız. Clock hattı APB1'e gidiyor.



### RCC APB1 peripheral clock enable register (RCC\_APB1ENR)

Address offset: 0x40

Reset value: 0x0000 0000

Access: no wait state, word, half-word and byte access.

| 31       | 30      | 29       | 28            | 27       | 26       | 25            | 24       | 23      | 22      | 21       | 20       | 19         | 18         | 17            | 16 |
|----------|---------|----------|---------------|----------|----------|---------------|----------|---------|---------|----------|----------|------------|------------|---------------|----|
| Reserved | DAC EN  | PWR EN   | Reser-<br>ved | CAN2 EN  | CAN1 EN  | Reser-<br>ved | I2C3 EN  | I2C2 EN | I2C1 EN | UART5 EN | UART4 EN | USART 3 EN | USART 2 EN | Reser-<br>ved |    |
| 15       | 14      | 13       | 12            | 11       | 10       | 9             | 8        | 7       | 6       | 5        | 4        | 3          | 2          | 1             | 0  |
| SPI3 EN  | SPI2 EN | Reserved | WWDG EN       | Reserved | TIM14 EN | TIM13 EN      | TIM12 EN | TIM7 EN | TIM6 EN | TIM5 EN  | TIM4 EN  | TIM3 EN    | TIM2 EN    |               |    |
| rw       | rw      |          | rw            |          | rw       | rw            | rw       | rw      | rw      | rw       | rw       | rw         | rw         |               |    |

18.biti 1 yapıyoruz.

Bit 18 **USART3EN**: USART3 clock enable

Set and cleared by software.

0: USART3 clock disabled

1: USART3 clock enabled

RCC->APB1ENR |= 1 << 18;

- USART3 hangi porta bağlı olduğunu öğrenmek için datasheet'e bakarız. B portun 10. ve 11.pinine bağlı olmuş.

| Port   | AF0  | AF1             | AF2       | AF3           | AF4        | AF5                     | AF6               | AF7                 |
|--------|------|-----------------|-----------|---------------|------------|-------------------------|-------------------|---------------------|
|        | SYS  | TIM1/2          | TIM3/4/5  | TIM8/9/10 /11 | I2C1/2/3   | SPI1/SPI2/ I2S2/I2S2ext | SPI3/I2Sext /I2S3 | USART1/2/3/ I2S3ext |
| Port B | PB0  | -               | TIM1_CH2N | TIM3_CH3      | TIM8_CH2N  | -                       | -                 | -                   |
|        | PB1  | -               | TIM1_CH3N | TIM3_CH4      | TIM8_CH3N  | -                       | -                 | -                   |
|        | PB2  | -               | -         | -             | -          | -                       | -                 | -                   |
|        | PB3  | JTDO/ TRACES_WO | TIM2_CH2  | -             | -          | -                       | SPI1_SCK          | SPI3_SCK I2S3_CK    |
|        | PB4  | NJTRST          | -         | TIM3_CH1      | -          | -                       | SPI1_MISO         | SPI3_MISO           |
|        | PB5  | -               | -         | TIM3_CH2      | I2C1_SMB_A | SPI1_MOSI               | SPI3_MOSI I2S3_SD |                     |
|        | PB6  | -               | -         | TIM4_CH1      | I2C1_SCL   | -                       | -                 | USART1_TX           |
|        | PB7  | -               | -         | TIM4_CH2      | I2C1_SDA   | -                       | -                 | USART1_RX           |
|        | PB8  | -               | -         | TIM4_CH3      | TIM10_CH1  | I2C1_SCL                | -                 | -                   |
|        | PB9  | -               | -         | TIM4_CH4      | TIM11_CH1  | I2C1_SDA                | SPI2 NSS I2S2_WS  | -                   |
|        | PB10 | -               | TIM2_CH3  | -             | -          | I2C2_SCL                | SPI2_SCK I2S2_CK  | -                   |
|        | PB11 | -               | TIM2_CH4  | -             | -          | I2C2_SDA                | -                 | USART3_RX           |
|        | PB12 | -               | TIM1_BKIN | -             | -          | I2C2_SMBA               | SPI2 NSS I2S2_WS  | -                   |
|        | PB13 | -               | TIM1_CH1N | -             | -          | -                       | SPI2_SCK I2S2_CK  | -                   |
|        | PB14 | -               | TIM1_CH2N | -             | TIM8_CH2N  | -                       | SPI2_MISO         | I2S2ext_SD          |
|        | PB15 | RTC_REFIN       | TIM1_CH3N | -             | TIM8_CH3N  | -                       | SPI2_MOSI I2S2_SD | -                   |

- Portlar AHB1 kısmına gidiyor.



## RCC AHB1 peripheral clock enable register (RCC\_AHB1ENR)

Address offset: 0x30

Reset value: 0x0010 0000

Access: no wait state, word, half-word and byte access.

- Biz B portunu kullandığımızdan sadece bunu aktif ediyoruz.

**Bit 1 GPIOBEN:** IO port B clock enable

This bit is set and cleared by software.

0: IO port B clock disabled

1: IO port B clock enabled

RCC->[AHB1ENR](#) |= 0x00000002;

- B portun 10. ve 11. pinleri USART3 için kullanacağımızdan bu pinleri alternate function mode yapıyoruz.

**GPIO port mode register (GPIOx\_MODER) (x = A..I/J/K)**

Address offset: 0x00

Reset values:

- 0xA800 0000 for port A
- 0x0000 0280 for port B
- 0x0000 0000 for other ports

| 31           | 30 | 29           | 28 | 27           | 26 | 25           | 24 | 23           | 22 | 21           | 20 | 19          | 18 | 17          | 16 |
|--------------|----|--------------|----|--------------|----|--------------|----|--------------|----|--------------|----|-------------|----|-------------|----|
| MODER15[1:0] |    | MODER14[1:0] |    | MODER13[1:0] |    | MODER12[1:0] |    | MODER11[1:0] |    | MODER10[1:0] |    | MODER9[1:0] |    | MODER8[1:0] |    |
| rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw          | rw | rw          | rw |
| 15           | 14 | 13           | 12 | 11           | 10 | 9            | 8  | 7            | 6  | 5            | 4  | 3           | 2  | 1           | 0  |
| MODER7[1:0]  |    | MODER6[1:0]  |    | MODER5[1:0]  |    | MODER4[1:0]  |    | MODER3[1:0]  |    | MODER2[1:0]  |    | MODER1[1:0] |    | MODER0[1:0] |    |
| rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw          | rw | rw          | rw |

Bits 2y:2y+1 **MODERY[1:0]:** Port x configuration bits (y = 0..15)

These bits are written by software to configure the I/O direction mode.

00: Input (reset state)

01: General purpose output mode

10: Alternate function mode

11: Analog mode

GPIOB->[MODER](#) |= 0x00A00000;

- Kullanacağımız çevresel birim olan USART3 kullanacağımızı belirteceğiz.

**GPIO alternate function low register (GPIOx\_AFRL) (x = A..I/J/K)**

Address offset: 0x20

Reset value: 0x0000 0000

| 31         | 30 | 29 | 28 | 27         | 26 | 25 | 24 | 23         | 22 | 21 | 20 | 19         | 18 | 17 | 16 |
|------------|----|----|----|------------|----|----|----|------------|----|----|----|------------|----|----|----|
| AFRL7[3:0] |    |    |    | AFRL6[3:0] |    |    |    | AFRL5[3:0] |    |    |    | AFRL4[3:0] |    |    |    |
| rw         | rw | rw | rw | rw         | rw | rw | rw | rw         | rw | rw | rw | rw         | rw | rw | rw |
| 15         | 14 | 13 | 12 | 11         | 10 | 9  | 8  | 7          | 6  | 5  | 4  | 3          | 2  | 1  | 0  |
| AFRL3[3:0] |    |    |    | AFRL2[3:0] |    |    |    | AFRL1[3:0] |    |    |    | AFRL0[3:0] |    |    |    |
| rw         | rw | rw | rw | rw         | rw | rw | rw | rw         | rw | rw | rw | rw         | rw | rw | rw |

**GPIO alternate function high register (GPIOx\_AFRH)**

(x = A..I/J)

Address offset: 0x24

Reset value: 0x0000 0000

| 31          | 30 | 29 | 28 | 27          | 26 | 25 | 24 | 23          | 22 | 21 | 20 | 19          | 18 | 17 | 16 |
|-------------|----|----|----|-------------|----|----|----|-------------|----|----|----|-------------|----|----|----|
| AFRH15[3:0] |    |    |    | AFRH14[3:0] |    |    |    | AFRH13[3:0] |    |    |    | AFRH12[3:0] |    |    |    |
| rw          | rw | rw | rw | rw          | rw | rw | rw | rw          | rw | rw | rw | rw          | rw | rw | rw |
| 15          | 14 | 13 | 12 | 11          | 10 | 9  | 8  | 7           | 6  | 5  | 4  | 3           | 2  | 1  | 0  |
| AFRH11[3:0] |    |    |    | AFRH10[3:0] |    |    |    | AFRH9[3:0]  |    |    |    | AFRH8[3:0]  |    |    |    |
| rw          | rw | rw | rw | rw          | rw | rw | rw | rw          | rw | rw | rw | rw          | rw | rw | rw |

- Biz 10 ve 11. pinleri kullandığımızdan high olanı kullanıyoruz.

Bits 31:0 **AFRH<sub>y</sub>:** Alternate function selection for port x bit y (y = 8..15)

These bits are written by software to configure alternate function I/Os

AFRH<sub>y</sub> selection:

|           |            |
|-----------|------------|
| 0000: AF0 | 1000: AF8  |
| 0001: AF1 | 1001: AF9  |
| 0010: AF2 | 1010: AF10 |
| 0011: AF3 | 1011: AF11 |
| 0100: AF4 | 1100: AF12 |
| 0101: AF5 | 1101: AF13 |
| 0110: AF6 | 1110: AF14 |
| 0111: AF7 | 1111: AF15 |

- Seçtiğimiz USART3 çevresel birimi hangi AF'de olduğunu bilmek için Reference Manuel kitapçığına bakıyoruz.

### Selecting an alternate function

For pins 0 to 7, the GPIOx\_AFRL[31:0] register selects the dedicated alternate function



For pins 8 to 15, the GPIOx\_AFRH[31:0] register selects the dedicated alternate function



ai17538

- USART3, AF7'de bulunuyor. Böylece pinlere AF7 için olan 0111 tanımlaması yapacağız.
- AFR'nin High ve Low olduğunda belirtmemiz gerekiyor. High kullandığımızdan 1 yazıyoruz.

`GPIOB->AFR[1] |= 7 << 8 | 7 << 12;`

- GPIO için yazdığımız fonksiyon aşağıdaki gibidir.

```
24 void GPIO_Config(void)
25 {
26     RCC->AHB1ENR |= 0x00000002; //B clock enable
27
28     GPIOB->MODER |= 0x00A00000; //PA10, PA11 Alternate function mode
29     GPIOB->AFR[1] |= (7 << 8) | (7 << 12); //USART3 AFRH10, AFRH11
30 }
```

### Baud rate register (USART\_BRR)

The baud counters stop counting if the TE or RE bits are disabled respectively.

Address offset: 0x08

Reset value: 0x0000 0000

| 31                 | 30 | 29 | 28 | 27 | 26 | 25 | 24 | 23 | 22 | 21 | 20 | 19                | 18 | 17 | 16 |
|--------------------|----|----|----|----|----|----|----|----|----|----|----|-------------------|----|----|----|
| Reserved           |    |    |    |    |    |    |    |    |    |    |    |                   |    |    |    |
| 15                 | 14 | 13 | 12 | 11 | 10 | 9  | 8  | 7  | 6  | 5  | 4  | 3                 | 2  | 1  | 0  |
| DIV_Mantissa[11:0] |    |    |    |    |    |    |    |    |    |    |    | DIV_Fraction[3:0] |    |    |    |
| rw                 | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw                | rw | rw | rw |

Bits 31:16 Reserved, must be kept at reset value

Bits 15:4 DIV\_Mantissa[11:0]: mantissa of USARTDIV

These 12 bits define the mantissa of the USART Divider (USARTDIV)

Bits 3:0 DIV\_Fraction[3:0]: fraction of USARTDIV

These 4 bits define the fraction of the USART Divider (USARTDIV). When OVER8=1, the DIV\_Fraction3 bit is not considered and must be kept cleared

## Baud rate register (USART\_BRR)

The baud counters stop counting if the TE or RE bits are disabled respectively.

Address offset: 0x08

Reset value: 0x0000 0000

|                    |    |    |    |    |    |    |    |    |    |    |    |                   |    |    |    |
|--------------------|----|----|----|----|----|----|----|----|----|----|----|-------------------|----|----|----|
| 31                 | 30 | 29 | 28 | 27 | 26 | 25 | 24 | 23 | 22 | 21 | 20 | 19                | 18 | 17 | 16 |
| Reserved           |    |    |    |    |    |    |    |    |    |    |    |                   |    |    |    |
| 15                 | 14 | 13 | 12 | 11 | 10 | 9  | 8  | 7  | 6  | 5  | 4  | 3                 | 2  | 1  | 0  |
| DIV_Mantissa[11:0] |    |    |    |    |    |    |    |    |    |    |    | DIV_Fraction[3:0] |    |    |    |
| rw                 | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw                | rw | rw | rw |

Bits 31:16 Reserved, must be kept at reset value

Bits 15:4 DIV\_Mantissa[11:0]: mantissa of USARTDIV

These 12 bits define the mantissa of the USART Divider (USARTDIV)

Bits 3:0 DIV\_Fraction[3:0]: fraction of USARTDIV

These 4 bits define the fraction of the USART Divider (USARTDIV). When OVER8=1, the DIV\_Fraction3 bit is not considered and must be kept cleared.

- Baud rate ayarını 9600 yapmak için BRR register'a 0x1112 yazıyoruz.

USART3->**BRR** |= 0x1112;

## Control register 1 (USART\_CR1)

Address offset: 0x0C

Reset value: 0x0000 0000

|          |          |    |    |      |     |    |      |       |      |        |        |    |    |     |     |
|----------|----------|----|----|------|-----|----|------|-------|------|--------|--------|----|----|-----|-----|
| 31       | 30       | 29 | 28 | 27   | 26  | 25 | 24   | 23    | 22   | 21     | 20     | 19 | 18 | 17  | 16  |
| Reserved |          |    |    |      |     |    |      |       |      |        |        |    |    |     |     |
| 15       | 14       | 13 | 12 | 11   | 10  | 9  | 8    | 7     | 6    | 5      | 4      | 3  | 2  | 1   | 0   |
| OVER8    | Reserved | UE | M  | WAKE | PCE | PS | PEIE | TXEIE | TCIE | RXNEIE | IDLEIE | TE | RE | RWU | SBK |
| rw       | Res.     | rw | rw | rw   | rw  | rw | rw   | rw    | rw   | rw     | rw     | rw | rw | rw  | rw  |

Bit 2 RE: Receiver enable

This bit enables the receiver. It is set and cleared by software.

0: Receiver is disabled

1: Receiver is enabled and begins searching for a start bit

USART3->**CR1** |= (1 << 2);

Bit 3 TE: Transmitter enable

This bit enables the transmitter. It is set and cleared by software.

0: Transmitter is disabled

1: Transmitter is enabled

Note: During transmission, a "0" pulse on the TE bit ("0" followed by "1") sends a preamble (idle line) after the current word, except in smartcard mode.

When TE is set, there is a 1 bit-time delay before the transmission starts.

USART3->**CR1** |= (1 << 3);

- Mesaj geldiğinde interrupt girmesi için aktif ediyoruz.

Bit 5 RXNEIE: RXNE interrupt enable

This bit is set and cleared by software.

0: Interrupt is inhibited

1: An USART interrupt is generated whenever ORE=1 or RXNE=1 in the USART\_SF register

USART3->**CR1** |= (1 << 5);

Bit 10 PCE: Parity control enable

This bit selects the hardware parity control (generation and detection). When the parity control is enabled, the computed parity is inserted at the MSB position (9th bit if M=1; 8th bit if M=0) and parity is checked on the received data. This bit is set and cleared by software. Once it is set, PCE is active after the current byte (in reception and in transmission).

0: Parity control disabled

1: Parity control enabled

USART3->**CR1** |= (0 << 10);

Bit 12 M: Word length

This bit determines the word length. It is set or cleared by software.

0: 1 Start bit, 8 Data bits, n Stop bit

1: 1 Start bit, 9 Data bits, n Stop bit

Note: The M bit must not be modified during a data transfer (both transmission and reception)

```
USART3->CR1 |= (0 << 12);
```

- USART fonksiyonun en alt satırında olması gerekiyor.

Bit 13 UE: USART enable

When this bit is cleared, the USART prescalers and outputs are stopped and the end of the current byte transfer in order to reduce power consumption. This bit is set and cleared by software.

0: USART prescaler and outputs disabled  
1: USART enabled

```
USART3->CR1 |= (1 << 13);
```

## Control register 2 (USART\_CR2)

Address offset: 0x10

Reset value: 0x0000 0000

| 31       | 30    | 29        | 28    | 27   | 26   | 25   | 24   | 23    | 22   | 21   | 20       | 19 | 18 | 17 | 16 |
|----------|-------|-----------|-------|------|------|------|------|-------|------|------|----------|----|----|----|----|
| Reserved |       |           |       |      |      |      |      |       |      |      |          |    |    |    |    |
| 15       | 14    | 13        | 12    | 11   | 10   | 9    | 8    | 7     | 6    | 5    | 4        | 3  | 2  | 1  | 0  |
| Res.     | LINEN | STOP[1:0] | CLKEN | CPOL | CPHA | LBCL | Res. | LBDIE | LBDL | Res. | ADD[3:0] |    |    |    |    |
|          | rw    | rw        | rw    | rw   | rw   | rw   |      | rw    | rw   | rw   | rw       | rw | rw | rw | rw |

Bits 13:12 STOP: STOP bits

These bits are used for programming the stop bits.

00: 1 Stop bit  
01: 0.5 Stop bit  
10: 2 Stop bits  
11: 1.5 Stop bit

Note: The 0.5 Stop bit and 1.5 Stop bit are not available for UART4 & UART5.

```
USART3->CR2 |= (0 << 12);
```

- USART fonksiyonu için ayarlamalar bitti.

```
35 void USART_Config()
36 {
37     RCC->APB1ENR |= 1 << 18;           //USART3EN
38
39     USART3->BRR |= 0x1112;             //BaudRate 9600
40     USART3->CR1 |= (1 << 2);         //Receiver Enable
41     USART3->CR1 |= (1 << 3);         //Transmitter Enable
42     USART3->CR1 |= (1 << 5);         //RXNE interrupt enable
43     USART3->CR1 |= (0 << 10);        //Parity control disabled
44     USART3->CR1 |= (0 << 12);        //Word length 8 Data bits
45     USART3->CR2 |= (0 << 12);        //1 Stop bit
46     USART3->CR1 |= (1 << 13);        //USART enable
47 }
```

## Kod Kısmı

- Şimdi interrupt için fonksiyon yazacağımız. Öncelikle NVIC için fonksiyon yazıyoruz.

## Status register (USART\_SR)

Address offset: 0x00

Reset value: 0x0000 00C0

| 31       | 30    | 29    | 28 | 27    | 26    | 25 | 24 | 23  | 22  | 21  | 20 | 19   | 18   | 17  | 16 |    |    |
|----------|-------|-------|----|-------|-------|----|----|-----|-----|-----|----|------|------|-----|----|----|----|
| Reserved |       |       |    |       |       |    |    |     |     |     |    |      |      |     |    |    |    |
| 15       | 14    | 13    | 12 | 11    | 10    | 9  | 8  | 7   | 6   | 5   | 4  | 3    | 2    | 1   | 0  |    |    |
| Reserved |       |       |    |       |       |    |    | CTS | LBD | TXE | TC | RXNE | IDLE | ORE | NF | FE | PE |
|          | rc_w0 | rc_w0 | r  | rc_w0 | rc_w0 | r  | r  | r   | r   | r   | r  | r    | r    | r   | r  | r  |    |

- 7.pini interrupt veriyoruz.

Bit 7 TXE: Transmit data register empty

This bit is set by hardware when the content of the TDR register has been transferred into the shift register. An interrupt is generated if the TXEIE bit =1 in the USART\_CR1 register. It is cleared by a write to the USART\_DR register.

0: Data is not transferred to the shift register  
1: Data is transferred to the shift register

Note: This bit is used during single buffer transmission.

```
NVIC->ISER[1] |= (1 << 7);
```

- Interrupt fonksiyonu yazıyoruz. Öncelikle değişken ataması yapıyoruz.  
Haberleşme durumunu Status Register ile bu değişkene yazacağız.

```
volatile int Str;
Str = USART3->SR;
```

```

49 void NVIC_Config(void)
50 {
51     NVIC->ISER[1] |= (1 << 7);           //Interrupt Set Enable Register
52 }

```

- Gelen mesajları dizeye alıyoruz. Bunun için Data register kullanıyoruz.

### Data register (USART\_DR)

Address offset: 0x04

Reset value: 0XXXXX XXXXX

Bits 31:9 Reserved, must be kept at reset value

Bits 8:0 DR[8:0]: Data value

Contains the Received or Transmitted data character, depending on whether it is read from or written to.

The Data register performs a double function (read and write) since it is composed of two registers, one for transmission (TDR) and one for reception (RDR)

The TDR register provides the parallel interface between the internal bus and the output shift register (see Figure 1).

The RDR register provides the parallel interface between the input shift register and the internal bus.

When transmitting with the parity enabled (PCE bit set to 1 in the USART\_CR1 register), the value written in the MSB (bit 7 or bit 8 depending on the data length) has no effect because it is replaced by the parity.

When receiving with the parity enabled, the value read in the MSB bit is the received parity bit.

```

3 char Rx_Buff[100];
4 int i=0;

58     Rx_Buff[i] = USART3->DR;
59     i++;

54 void USART3_IRQHandler()
55 {
56     volatile int Str;
57     Str = USART3->SR;
58     Rx_Buff[i] = USART3->DR;
59     i++;
60 }

62 void Send_Char(char message)
63 {
64     while(!(USART3->SR & 1 << 7));
65     USART3->DR = message;
66 }

68 void Send_Message(char *Str)
69 {
70     while(*Str)
71     {
72         Send_Char(*Str);
73         Str++;
74     }
75 }

77 int main(void)
78 {
79     RCC_Config();
80     GPIO_Config();
81     USART_Config();
82     NVIC_Config();
83
84     while (1)
85     {
86         Send_Message("Hello World \n");
87         for(int i=0; i<1000000; i++);
88     }
89 }

```

- Send\_Message'a mesajımızı yazıyoruz. Mesajımızdaki harflerin yanı her dizideki elemanlarını alıyoruz ve her elemanı Send\_Char'a gönderiyoruz ve burada Data register'a yazarak karşı tarafa mesaj ulaşıyor

# 09\_01 I2C Kullanımı

25 Aralık 2021 Cumartesi 01:02

## 09\_01 I2C Kullanımı

### ➤ REGISTER

#### Konfigürasyon Kısmı

- RCC için yazdığımız fonksiyon aşağıdaki gibidir.

```
3 void RCC_Config(void)
4 {
5     RCC->CR |= 0x00010000; //HSEON
6     while(!(RCC->CR & 0x00020000)); //HSERDY
7     RCC->CR |= 0x00080000; //CSSON
8     RCC->CFGR = 0x00000000;
9     RCC->PLLCFGR |= 0x00400000; //PLLSRC
10    RCC->PLLCFGR |= 0x00000004; //PLLM 4
11    RCC->PLLCFGR |= 0x00002A00; //PLLN 168
12    RCC->PLLCFGR |= 0x00000000; //PLLP 2
13    RCC->CR |= 0x01000000; //PLLON
14    while(!(RCC->CR & 0x02000000)); //PLLRDY
15    RCC->CFGR |= 0x00000001; //SW
16    while(!(RCC->CR & 0x00000001)); //SWS
17    RCC->CFGR |= 0x00000000; //HPRE AHB 1
18    RCC->CFGR |= 0x00001400; //PPRE1 APB1 4
19    RCC->CFGR |= 0x00008000; //PPRE2 APB2 2
20    RCC->CIR |= 0x00080000; //HSERDYC
21    RCC->CIR |= 0x00800000; //CSSC
22 }
```

- I2C2 kullanacağımız. Clock hattı APB1'e gidiyor





### RCC APB1 peripheral clock enable register (RCC\_APB1ENR)

Address offset: 0x40

Reset value: 0x0000 0000

Access: no wait state, word, half-word and byte access.

| 31       | 30      | 29       | 28            | 27       | 26       | 25            | 24       | 23       | 22      | 21       | 20       | 19         | 18         | 17            | 16 |
|----------|---------|----------|---------------|----------|----------|---------------|----------|----------|---------|----------|----------|------------|------------|---------------|----|
| Reserved | DAC EN  | PWR EN   | Reser-<br>ved | CAN2 EN  | CAN1 EN  | Reser-<br>ved | I2C3 EN  | I2C2 EN  | I2C1 EN | UART5 EN | UART4 EN | USART 3 EN | USART 2 EN | Reser-<br>ved |    |
|          | rw      | rw       |               | rw       | rw       |               | rw       | rw       | rw      | rw       | rw       | rw         | rw         |               |    |
| 15       | 14      | 13       | 12            | 11       | 10       | 9             | 8        | 7        | 6       | 5        | 4        | 3          | 2          | 1             | 0  |
| SPI3 EN  | SPI2 EN | Reserved | WWDG EN       | Reserved | Reserved | TIM14 EN      | TIM13 EN | TIM12 EN | TIM7 EN | TIM6 EN  | TIM5 EN  | TIM4 EN    | TIM3 EN    | TIM2 EN       |    |
| rw       | rw      |          | rw            |          |          | rw            | rw       | rw       | rw      | rw       | rw       | rw         | rw         | rw            |    |

22.biti 1 yapıyoruz.

Bit 22 I2C2EN: I2C2 clock enable

This bit is set and cleared by software.

0: I2C2 clock disabled

1: I2C2 clock enabled

RCC->[APB1ENR](#) |= 1 << 22;

- I2C2 hangi porta bağlı olduğunu öğrenmek için datasheet'e bakarız. B portun 10. ve 11.pinine bağlıymış.

| Port   |      | AF0                   | AF1       | AF2      | AF3           | AF4        | AF5                     | AF6                  | AF7                 |
|--------|------|-----------------------|-----------|----------|---------------|------------|-------------------------|----------------------|---------------------|
|        |      | SYS                   | TIM1/2    | TIM3/4/5 | TIM8/9/10 /11 | I2C1/2/3   | SPI1/SPI2/ I2S2/I2S2ext | SPI3/I2Sext /I2S3    | USART1/2/3/ I2S3ext |
| Port B | PB0  | -                     | TIM1_CH2N | TIM3_CH3 | TIM8_CH2N     | -          | -                       | -                    | -                   |
|        | PB1  | -                     | TIM1_CH3N | TIM3_CH4 | TIM8_CH3N     |            | -                       | -                    | -                   |
|        | PB2  | -                     | -         | -        | -             | -          | -                       | -                    | -                   |
|        | PB3  | JTDO/<br>TRACES<br>WO | TIM2_CH2  | -        | -             | -          | SPI1_SCK<br>I2S3_CK     |                      | -                   |
|        | PB4  | NJTRST                | -         | TIM3_CH1 |               | -          | SPI1_MISO               | SPI3_MISO            | I2S3ext_SD          |
|        | PB5  | -                     | -         | TIM3_CH2 |               | I2C1_SMB_A | SPI1莫斯I                 | SPI3_MOSI<br>I2S3_SD |                     |
|        | PB6  | -                     | -         | TIM4_CH1 |               | I2C1_SCL   | -                       | -                    | USART1_TX           |
|        | PB7  | -                     | -         | TIM4_CH2 |               | I2C1_SDA   | -                       | -                    | USART1_RX           |
|        | PB8  | -                     | -         | TIM4_CH3 | TIM10_CH1     | I2C1_SCL   | -                       | -                    | -                   |
|        | PB9  | -                     | -         | TIM4_CH4 | TIM11_CH1     | I2C1_SDA   | SPI2_NSS<br>I2S2_WS     | -                    | -                   |
|        | PB10 | -                     | TIM2_CH3  | -        | -             | I2C2_SCL   | SPI2_SCK<br>I2S2_CK     | -                    | USART3_TX           |
|        | PB11 | -                     | TIM2_CH4  | -        | -             | I2C2_SDA   | -                       | -                    | USART3_RX           |
|        | PB12 | -                     | TIM1_BKIN | -        | -             | I2C2_SMBA  | SPI2_NSS<br>I2S2_WS     | -                    | USART3_CK           |
|        | PB13 | -                     | TIM1_CH1N | -        | -             | -          | SPI2_SCK<br>I2S2_CK     | -                    | USART3_CTS          |
|        | PB14 | -                     | TIM1_CH2N | -        | TIM8_CH2N     | -          | SPI2_MISO               | I2S2ext_SD           | USART3_RTS          |
|        | PB15 | RTC_REFIN             | TIM1_CH3N | -        | TIM8_CH3N     | -          | SPI2_MOSI<br>I2S2_SD    | -                    | -                   |

- Portlar AHB1 kısmına gidiyor.



### RCC AHB1 peripheral clock enable register (RCC\_AHB1ENR)

Address offset: 0x30

Reset value: 0x0010 0000

Access: no wait state, word, half-word and byte access.

| 31            | 30                      | 29          | 28                  | 27                 | 26                 | 25           | 24         | 23          | 22          | 21               | 20      | 19            | 18          | 17          | 16          |    |  |  |  |  |  |  |
|---------------|-------------------------|-------------|---------------------|--------------------|--------------------|--------------|------------|-------------|-------------|------------------|---------|---------------|-------------|-------------|-------------|----|--|--|--|--|--|--|
| Reser-<br>ved | OTGH<br>S<br>ULPIE<br>N | OTGH<br>SEN | ETHM<br>ACPTP<br>EN | ETHM<br>ACRXE<br>N | ETHM<br>ACTXE<br>N | ETHMA<br>CEN | Reserved   | DMA2E<br>N  | DMA1E<br>N  | CCMDAT<br>ARAMEN | Res.    | BKPSR<br>AMEN | Reserved    |             |             |    |  |  |  |  |  |  |
|               | rw                      | rw          | rw                  | rw                 | rw                 | rw           |            | rw          | rw          |                  |         | rw            |             |             |             |    |  |  |  |  |  |  |
| 15            | 14                      | 13          | 12                  | 11                 | 10                 | 9            |            |             |             |                  |         |               |             |             |             |    |  |  |  |  |  |  |
|               |                         |             |                     |                    |                    | Reserved     | GPIOE<br>N | GPIOH<br>EN | GPIOG<br>EN | GPIOF<br>N       | GPIOEEN | GPIOD<br>EN   | GPIOC<br>EN | GPIO<br>BEN | GPIO<br>AEN |    |  |  |  |  |  |  |
| Reserved      |                         |             |                     |                    |                    |              |            | rw          | rw          | rw               | rw      | rw            | rw          | rw          | rw          | rw |  |  |  |  |  |  |
|               |                         |             |                     |                    |                    |              |            |             |             |                  |         |               |             |             |             |    |  |  |  |  |  |  |

- Buton için A portunu, I2C2 için B portunu kullandığımızdan bu portları aktif ediyoruz.

Bit 0 **GPIOAEN**: IO port A clock enable

This bit is set and cleared by software.

0: IO port A clock disabled

1: IO port A clock enabled

Bit 1 **GPIOBEN**: IO port B clock enable

This bit is set and cleared by software.

0: IO port B clock disabled

1: IO port B clock enabled

RCC->[AHB1ENR](#) |= 0x00000002;

- B portun 10. ve 11. pinleri I2C için kullanacağımızdan bu pinleri alternate function mode yapıyoruz.

## GPIO port mode register (GPIOx\_MODER) (x = A..I/J/K)

Address offset: 0x00

Reset values:

- 0xA800 0000 for port A
- 0x0000 0280 for port B
- 0x0000 0000 for other ports

| 31           | 30 | 29           | 28 | 27           | 26 | 25           | 24 | 23           | 22 | 21           | 20 | 19          | 18 | 17          | 16 |
|--------------|----|--------------|----|--------------|----|--------------|----|--------------|----|--------------|----|-------------|----|-------------|----|
| MODER15[1:0] |    | MODER14[1:0] |    | MODER13[1:0] |    | MODER12[1:0] |    | MODER11[1:0] |    | MODER10[1:0] |    | MODER9[1:0] |    | MODER8[1:0] |    |
| rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw          | rw | rw          | rw |
| 15           | 14 | 13           | 12 | 11           | 10 | 9            | 8  | 7            | 6  | 5            | 4  | 3           | 2  | 1           | 0  |
| MODER7[1:0]  |    | MODER6[1:0]  |    | MODER5[1:0]  |    | MODER4[1:0]  |    | MODER3[1:0]  |    | MODER2[1:0]  |    | MODER1[1:0] |    | MODER0[1:0] |    |
| rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw           | rw | rw          | rw | rw          | rw |

Bits 2y:2y+1 MODERy[1:0]: Port x configuration bits (y = 0..15)

These bits are written by software to configure the I/O direction mode.

00: Input (reset state)

01: General purpose output mode

10: Alternate function mode

11: Analog mode

GPIOB->MODER |= (2 << 20) | (2 << 22) ;

- Kullanacağımız çevresel birim olan I2C kullanacağımızı belirteceğiz.

## GPIO alternate function low register (GPIOx\_AFRL) (x = A..I/J/K)

Address offset: 0x20

Reset value: 0x0000 0000

| 31         | 30 | 29 | 28 | 27         | 26 | 25 | 24 | 23         | 22 | 21 | 20 | 19         | 18 | 17 | 16 |
|------------|----|----|----|------------|----|----|----|------------|----|----|----|------------|----|----|----|
| AFRL7[3:0] |    |    |    | AFRL6[3:0] |    |    |    | AFRL5[3:0] |    |    |    | AFRL4[3:0] |    |    |    |
| rw         | rw | rw | rw | rw         | rw | rw | rw | rw         | rw | rw | rw | rw         | rw | rw | rw |
| 15         | 14 | 13 | 12 | 11         | 10 | 9  | 8  | 7          | 6  | 5  | 4  | 3          | 2  | 1  | 0  |
| AFRL3[3:0] |    |    |    | AFRL2[3:0] |    |    |    | AFRL1[3:0] |    |    |    | AFRL0[3:0] |    |    |    |
| rw         | rw | rw | rw | rw         | rw | rw | rw | rw         | rw | rw | rw | rw         | rw | rw | rw |

## GPIO alternate function high register (GPIOx\_AFRH)

(x = A..I/J)

Address offset: 0x24

Reset value: 0x0000 0000

| 31          | 30 | 29 | 28 | 27          | 26 | 25 | 24 | 23          | 22 | 21 | 20 | 19          | 18 | 17 | 16 |
|-------------|----|----|----|-------------|----|----|----|-------------|----|----|----|-------------|----|----|----|
| AFRH15[3:0] |    |    |    | AFRH14[3:0] |    |    |    | AFRH13[3:0] |    |    |    | AFRH12[3:0] |    |    |    |
| rw          | rw | rw | rw | rw          | rw | rw | rw | rw          | rw | rw | rw | rw          | rw | rw | rw |
| 15          | 14 | 13 | 12 | 11          | 10 | 9  | 8  | 7           | 6  | 5  | 4  | 3           | 2  | 1  | 0  |
| AFRH11[3:0] |    |    |    | AFRH10[3:0] |    |    |    | AFRH9[3:0]  |    |    |    | AFRH8[3:0]  |    |    |    |
| rw          | rw | rw | rw | rw          | rw | rw | rw | rw          | rw | rw | rw | rw          | rw | rw | rw |

- Biz 10 ve 11. pinleri kullandığımızdan high olanı kullanıyoruz.

Bits 31:0 **AFRH<sub>y</sub>**: Alternate function selection for port x bit y (y = 8..15)

These bits are written by software to configure alternate function I/Os

AFRH<sub>y</sub> selection:

|           |            |
|-----------|------------|
| 0000: AF0 | 1000: AF8  |
| 0001: AF1 | 1001: AF9  |
| 0010: AF2 | 1010: AF10 |
| 0011: AF3 | 1011: AF11 |
| 0100: AF4 | 1100: AF12 |
| 0101: AF5 | 1101: AF13 |
| 0110: AF6 | 1110: AF14 |
| 0111: AF7 | 1111: AF15 |

- Seçtiğimiz I2C2 çevresel birimi hangi AF'de olduğunu bilmek için Reference Manuel kitapçığına bakıyoruz.

## Selecting an alternate function

For pins 0 to 7, the GPIOx\_AFRL[31:0] register selects the dedicated alternate function



For pins 8 to 15, the GPIOx\_AFRH[31:0] register selects the dedicated alternate function



ai17538

- I2C, AF7'de bulunuyor. Böylece pinlere AF4 için olan 0100 tanımlaması yapacağız.
- AFR'nin High ve Low olduğunda belirtmemiz gerekiyor. High kullandığımızdan 1 yazıyoruz.

GPIOB->AFR[1] |= (4 << 8) | (4 << 12);

## GPIO port output type register (GPIOx\_OTYPER) (x = A..I/J/K)

Address offset: 0x04

Reset value: 0x0000 0000

| 31       | 30   | 29   | 28   | 27   | 26   | 25  | 24  | 23  | 22  | 21  | 20  | 19  | 18  | 17  | 16  |
|----------|------|------|------|------|------|-----|-----|-----|-----|-----|-----|-----|-----|-----|-----|
| Reserved |      |      |      |      |      |     |     |     |     |     |     |     |     |     |     |
| 15       | 14   | 13   | 12   | 11   | 10   | 9   | 8   | 7   | 6   | 5   | 4   | 3   | 2   | 1   | 0   |
| OT15     | OT14 | OT13 | OT12 | OT11 | OT10 | OT9 | OT8 | OT7 | OT6 | OT5 | OT4 | OT3 | OT2 | OT1 | OT0 |
| rw       | rw   | rw   | rw   | rw   | rw   | rw  | rw  | rw  | rw  | rw  | rw  | rw  | rw  | rw  | rw  |

Bits 15:0 **OTy**: Port x configuration bits (y = 0..15)

These bits are written by software to configure the output type of the I/O port.

0: Output push-pull (reset state)

1: Output open-drain

```
GPIOB->OTYPER |= (1 << 10) | (1 << 11);
```

- GPIO için yazdığımız fonksiyon aşağıdaki gibidir.

```
24 void GPIO_Config()
25 {
26     RCC->AHB1ENR |= (3 << 0); //A, B clock enable
27
28     GPIOB->MODER |= (2 << 20) | (2 << 22); //PB10, PB11 Alternate function mode
29     GPIOB->AFR[1] |= (4 << 8) | (4 << 12); //I2C2 AFRH10, AFRH11
30     GPIOB->OTYPER |= (1 << 10) | (1 << 11); //Output open-drain
31 }
```

## I<sup>2</sup>C Control register 2 (I2C\_CR2)

Address offset: 0x04

Reset value: 0x0000

| 15       | 14 | 13   | 12    | 11      | 10      | 9       | 8        | 7  | 6         | 5  | 4  | 3  | 2  | 1  | 0  |  |
|----------|----|------|-------|---------|---------|---------|----------|----|-----------|----|----|----|----|----|----|--|
| Reserved |    | LAST | DMAEN | ITBUFEN | ITEVTEN | ITERREN | Reserved |    | FREQ[5:0] |    |    |    |    |    |    |  |
| rw       | rw | rw   | rw    | rw      | rw      | rw      | rw       | rw | rw        | rw | rw | rw | rw | rw | rw |  |

Bits 5:0 **FREQ[5:0]**: Peripheral clock frequency

The FREQ bits must be configured with the APB clock frequency value (I2C peripheral connected to APB). The FREQ field is used by the peripheral to generate data setup and hold times compliant with the I2C specifications. The minimum allowed frequency is 2 MHz, the maximum frequency is limited by the maximum APB frequency and cannot exceed 50 MHz (peripheral intrinsic maximum limit).

0b000000: Not allowed

0b000001: Not allowed

0b000010: 2 MHz

...

0b110010: 50 MHz

Higher than 0b100100: Not allowed

```
I2C2->CR2 |= 0x0008;
```

## I<sup>2</sup>C Clock control register (I<sup>2</sup>C\_CCR)

Address offset: 0x1C

Reset value: 0x0000

$f_{PCLK1}$  must be at least 2 MHz to achieve Sm mode I<sup>2</sup>C frequencies. It must be at least 4 MHz to achieve Fm mode I<sup>2</sup>C frequencies. It must be a multiple of 10MHz to reach the 400 kHz maximum I<sup>2</sup>C Fm mode clock.

The CCR register must be configured only when the I<sup>2</sup>C is disabled (PE = 0).

| 15  | 14   | 13       | 12 | 11        | 10 | 9  | 8  | 7  | 6  | 5  | 4  | 3  | 2  | 1  | 0  |
|-----|------|----------|----|-----------|----|----|----|----|----|----|----|----|----|----|----|
| F/S | DUTY | Reserved |    | CCR[11:0] |    |    |    |    |    |    |    |    |    |    |    |
| rw  | rw   |          |    | rw        | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw | rw |

Bits 11:0 CCR[11:0]: Clock control register in Fm/Sm mode (Master mode)

Controls the SCL clock in master mode.

Sm mode or SMBus:

$$T_{high} = CCR * T_{PCLK1}$$

$$T_{low} = CCR * T_{PCLK1}$$

Fm mode:

If DUTY = 0:

$$T_{high} = CCR * T_{PCLK1}$$

$$T_{low} = 2 * CCR * T_{PCLK1}$$

If DUTY = 1: (to reach 400 kHz)

$$T_{high} = 9 * CCR * T_{PCLK1}$$

$$T_{low} = 16 * CCR * T_{PCLK1}$$

For instance: in Sm mode, to generate a 100 kHz SCL frequency:

If FREQR = 08,  $T_{PCLK1} = 125$  ns so CCR must be programmed with 0x28  
(0x28 => 40d x 125 ns = 5000 ns.)

Note: The minimum allowed value is 0x04, except in FAST DUTY mode where the minimum allowed value is 0x01

$t_{high} = t_{r(SCL)} + t_{w(SCLH)}$ . See device datasheet for the definitions of parameters.

$t_{low} = t_{f(SCL)} + t_{w(SCLL)}$ . See device datasheet for the definitions of parameters.

I<sup>2</sup>C communication speed,  $f_{SCL} \sim 1/(t_{high} + t_{low})$ . The real frequency may differ due to the analog noise filter input delay.

The CCR register must be configured only when the I<sup>2</sup>C is disabled (PE = 0).

- 100 kHz çalışmak için 0x28 yazınız demiş.

I<sup>2</sup>C2->[CR2](#) |= 0x0008;

## I<sup>2</sup>C TRISE register (I<sup>2</sup>C\_TRISE)

Address offset: 0x20

Reset value: 0x0002

| 15       | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3          | 2  | 1  | 0  |
|----------|----|----|----|----|----|---|---|---|---|---|---|------------|----|----|----|
| Reserved |    |    |    |    |    |   |   |   |   |   |   | TRISE[5:0] |    |    |    |
|          |    |    |    |    |    |   |   |   |   |   |   | rw         | rw | rw | rw |

Bits 15:6 Reserved, must be kept at reset value

Bits 5:0 **TRISE[5:0]**: Maximum rise time in Fm/Sm mode (Master mode)

These bits should provide the maximum duration of the SCL feedback loop in master mode. The purpose is to keep a stable SCL frequency whatever the SCL rising edge duration. These bits must be programmed with the maximum SCL rise time given in the I<sup>2</sup>C bus specification, incremented by 1.

For instance: in Sm mode, the maximum allowed SCL rise time is 1000 ns.

If, in the I2C\_CR2 register, the value of FREQ[5:0] bits is equal to 0x08 and T<sub>PCLK1</sub> = 125 ns therefore the TRISE[5:0] bits must be programmed with 09h.

(1000 ns / 125 ns = 8 + 1)

The filter value can also be added to TRISE[5:0].

If the result is not an integer, TRISE[5:0] must be programmed with the integer part, in order to respect the t<sub>HIGH</sub> parameter.

*Note: TRISE[5:0] must be configured only when the I2C is disabled (PE = 0).*

- Geri besleme döngüsünün maksimum süresi için I2C\_CR2'nin FREQ[5:0] bitlerine 0x08 yazdığımızdan TRISE[5:0] bitlerine 0x09 yazmalıyız.

I2C2->**TRISE** |= 0x09;

## I<sup>2</sup>C Control register 1 (I2C\_CR1)

Address offset: 0x00

Reset value: 0x0000

| 15    | 14   | 13    | 12  | 11  | 10  | 9    | 8     | 7          | 6    | 5     | 4     | 3        | 2    | 1     | 0  |
|-------|------|-------|-----|-----|-----|------|-------|------------|------|-------|-------|----------|------|-------|----|
| SWRST | Res. | ALERT | PEC | POS | ACK | STOP | START | NO STRETCH | ENGC | ENPEC | ENARP | SMB TYPE | Res. | SMBUS | PE |
| rw    |      | rw    | rw  | rw  | rw  | rw   | rw    | rw         | rw   | rw    | rw    | rw       |      | rw    | rw |

Bit 0 **PE**: Peripheral enable

0: Peripheral disable

1: Peripheral enable

*Note: If this bit is reset while a communication is on going, the peripheral is disabled at the end of the current communication, when back to IDLE state.*

*All bit resets due to PE=0 occur at the end of the communication.*

*In master mode, this bit must not be reset before the end of the communication.*

- I2C2 aktif edildi.

I2C2->**CR1** |= (1 << 0);

I2C için yazdığımız fonksiyon aşağıdaki gibidir.

```
33 void I2C_Config()
34 {
35     RCC->APB1ENR |= 1 << 22;           //I2CEN
36
37     I2C2->CR2 |= 0x0008;                //8MHz
38     I2C2->CCR |= 0x0028;                //100kHz
39     I2C2->TRISE |= 0x09;                //Maximum rise time
40     I2C2->CR1 |= (1 << 0);            //Peripheral enable
41 }
```

## Kod Kısmı

- I2C için Write ve Read fonksiyonlarını yazmaya başlıyoruz.
- I2C\_Write fonksiyonuna adres ve data olmak üzere 2 parametre yazıyoruz.

Bit 8 **START**: Start generation

This bit is set and cleared by software and cleared by hardware when start is sent or PE=0.

In Master Mode:

0: No Start generation

1: Repeated start generation

In Slave mode:

0: No Start generation

1: Start generation when the bus is free

- Start yolluyorum.

I2C2->**CR1** |= (1 << 8);

## I<sup>2</sup>C Status register 1 (I2C\_SR1)

Address offset: 0x14

Reset value: 0x0000

| 15        | 14       | 13   | 12      | 11    | 10    | 9     | 8     | 7   | 6    | 5    | 4     | 3     | 2   | 1    | 0  |
|-----------|----------|------|---------|-------|-------|-------|-------|-----|------|------|-------|-------|-----|------|----|
| SMB ALERT | TIME OUT | Res. | PEC ERR | OVR   | AF    | ARLO  | BERR  | TxE | RxNE | Res. | STOPF | ADD10 | BTF | ADDR | SB |
| rc_w0     | rc_w0    |      | rc_w0   | rc_w0 | rc_w0 | rc_w0 | rc_w0 | r   | r    |      | r     | r     | r   | r    | r  |

- Yollandığım startı kontrol ediyorum.

Bit 0 **SB**: Start bit (Master mode)

0: No Start condition

1: Start condition generated.

– Set when a Start condition generated.

– Cleared by software by reading the SR1 register followed by writing the DR register, or by hardware when PE=0

- While döngüsü 1 olduğu sürece çalışır.
- SB biti 1 olana dek 0 olacağından ve döngünün çalışabilmesi için başına "!" işaretini koyarak tersliyoruz.
- 0.bit 1 olduğunda döngüden çıkışacaktır.

**while**(!(I2C2->**SR1** & (1 << 0)));

- Slave tarafına adresini yolluyoruz. Bizim kullandığımız cihazın adresi 0x4E'dir.

## I<sup>2</sup>C Data register (I2C\_DR)

Address offset: 0x10

Reset value: 0x0000

| 15       | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7       | 6  | 5  | 4  | 3  | 2  | 1  | 0  |
|----------|----|----|----|----|----|---|---|---------|----|----|----|----|----|----|----|
| Reserved |    |    |    |    |    |   |   | DR[7:0] |    |    |    |    |    |    |    |
|          |    |    |    |    |    |   |   | rw      | rw | rw | rw | rw | rw | rw | rw |

Bits 7:0 **DR[7:0]** 8-bit data register

Byte received or to be transmitted to the bus.

– Transmitter mode: Byte transmission starts automatically when a byte is written in the DR register. A continuous transmit stream can be maintained if the next data to be transmitted is put in DR once the transmission is started (TxE=1)

– Receiver mode: Received byte is copied into DR (RxNE=1). A continuous transmit stream can be maintained if DR is read before the next data byte is received (RxNE=1).

*Note: In slave mode, the address is not copied into DR.*

*Write collision is not managed (DR can be written if TxE=0).*

*If an ARLO event occurs on ACK pulse, the received byte is not copied into DR and so cannot be read.*



## I<sup>2</sup>C Status register 1 (I2C\_SR1)

Address offset: 0x14

Reset value: 0x0000

| 15        | 14       | 13   | 12      | 11    | 10    | 9     | 8     | 7   | 6    | 5    | 4     | 3     | 2   | 1    | 0  |
|-----------|----------|------|---------|-------|-------|-------|-------|-----|------|------|-------|-------|-----|------|----|
| SMB ALERT | TIME OUT | Res. | PEC ERR | OVR   | AF    | ARLO  | BERR  | TxE | RxNE | Res. | STOPF | ADD10 | BTF | ADDR | SB |
| rc_w0     | rc_w0    |      | rc_w0   | rc_w0 | rc_w0 | rc_w0 | rc_w0 | r   | r    |      | r     | r     | r   | r    | r  |

Bit 7 **TxE**: Data register empty (transmitters)

0: Data register not empty

1: Data register empty

- Set when DR is empty in transmission. TxE is not set during address phase.

- Cleared by software writing to the DR register or by hardware after a start or a stop condition or when PE=0.

TxE is not set if either a NACK is received, or if next byte to be transmitted is PEC (PEC=1)

*Note: TxE is not cleared by writing the first data being transmitted, or by writing data when BTF is set, as in both cases the data register is still empty.*

```
while(!(I2C2->SR2 & (1 << 7)));
```

```
I2C2->DR = data;
```

## I<sup>2</sup>C Status register 1 (I2C\_SR1)

Address offset: 0x14

Reset value: 0x0000

| 15        | 14       | 13   | 12      | 11    | 10    | 9     | 8     | 7   | 6    | 5    | 4     | 3     | 2   | 1    | 0  |
|-----------|----------|------|---------|-------|-------|-------|-------|-----|------|------|-------|-------|-----|------|----|
| SMB ALERT | TIME OUT | Res. | PEC ERR | OVR   | AF    | ARLO  | BERR  | TxE | RxNE | Res. | STOPF | ADD10 | BTF | ADDR | SB |
| rc_w0     | rc_w0    |      | rc_w0   | rc_w0 | rc_w0 | rc_w0 | rc_w0 | r   | r    |      | r     | r     | r   | r    | r  |

Bit 2 **BTF**: Byte transfer finished

0: Data byte transfer not done

1: Data byte transfer succeeded

- Set by hardware when NOSTRETCH=0 and:

- In reception when a new byte is received (including ACK pulse) and DR has not been read yet (RxNE=1).

- In transmission when a new byte should be sent and DR has not been written yet (TxE=1).

- Cleared by software by either a read or write in the DR register or by hardware after a start or a stop condition in transmission or when PE=0.

*Note: The BTF bit is not set after a NACK reception*

*The BTF bit is not set if next byte to be transmitted is the PEC (TRA=1 in I2C\_SR2 register and PEC=1 in I2C\_CR1 register)*

```
while(!(I2C2->SR2 & (1 << 2)));
```

## I<sup>2</sup>C Control register 1 (I2C\_CR1)

Address offset: 0x00

Reset value: 0x0000

| 15    | 14   | 13    | 12  | 11  | 10  | 9    | 8     | 7          | 6    | 5     | 4     | 3        | 2    | 1     | 0  |
|-------|------|-------|-----|-----|-----|------|-------|------------|------|-------|-------|----------|------|-------|----|
| SWRST | Res. | ALERT | PEC | POS | ACK | STOP | START | NO STRETCH | ENGC | ENPEC | ENARP | SMB TYPE | Res. | SMBUS | PE |
| rw    |      | rw    | rw  | rw  | rw  | rw   | rw    | rw         | rw   | rw    | rw    | rw       |      | rw    | rw |

Bit 9 **STOP**: Stop generation

The bit is set and cleared by software, cleared by hardware when a Stop condition is detected, set by hardware when a timeout error is detected.

In Master Mode:

0: No Stop generation.

1: Stop generation after the current byte transfer or after the current Start condition is sent.

In Slave mode:

0: No Stop generation.

1: Release the SCL and SDA lines after the current byte transfer.

```
I2C2->CR1 |= (1 << 9);
```

- I2C için yazdığımız Write fonksiyon aşağıdaki gibidir.

```
43 void I2C_Write(uint8_t adress, uint8_t data)
44 {
45     I2C2->CR1 |= (1 << 8);                      //Start generation
46     while(!(I2C2->SR1 & (1 << 0)));           //Start bit
47     I2C2->DR = 0x4E;                            //Slave adress
48     while(!(I2C2->SR1 & (1 << 1)));           //Received address matched
49     while(!(I2C2->SR2 & (1 << 0)));           //Master Mode
50     //I2C2->DR = adress;
51     while(!(I2C2->SR2 & (1 << 0)));           //Master Mode
52     while(!(I2C2->SR2 & (1 << 7)));           //Data register empty
53     I2C2->DR = data;                           //Data byte transfer succeeded
54     while(!(I2C2->SR2 & (1 << 2)));           //Data byte transfer succeeded
55     I2C2->CR1 |= (1 << 9);                   //Stop generation
56 }
```

- Her butona bastığımızda cihaza sırayla adres yolluyor.

```
65 int main(void)
66 {
67     RCC_Config();
68     GPIO_Config();
69     I2C_Config();
70
71     while (1)
72     {
73         if(GPIOA->IDR & 0x00000001)
74         {
75             i++;
76             delay(6300000);
77         }
78
79         switch(i)
80         {
81             case 0:
82                 I2C_Write(m_address, 0x00);
83                 break;
84             case 1:
85                 I2C_Write(m_address, 0x01);
86                 break;
87             case 2:
88                 I2C_Write(m_address, 0x02);
89                 break;
90             case 3:
91                 I2C_Write(m_address, 0x04);
92                 break;
93             case 4:
94                 I2C_Write(m_address, 0x08);
95                 break;
96             case 5:
97                 I2C_Write(m_address, 0x10);
98                 break;
99         }
100    }
```

```
96     case 5:  
97         I2C_Write(m_address, 0x10);  
98         break;  
99     case 6:  
100        I2C_Write(m_address, 0x20);  
101        break;  
102    case 7:  
103        I2C_Write(m_address, 0x40);  
104        break;  
105    case 8:  
106        I2C_Write(m_address, 0x80);  
107        break;  
108    default:  
109        i=0;  
110        break;  
111    }  
112  
113 }  
114 }
```

10 01 SPI Kullanımı

14 Mayıs 2022 Cumartesi 11:49

10\_01 SPI Kullanımı

➤ HAL

## Teori

- <https://deepbluembedded.com/stm32-spi-tutorial/> linkinden konu hakkındaki bilgileri inceleyebiliriz.
  - SPI haberleşmesi için üç farklı mod vardır. İki çift yönlü iken diğer tek yönlü haberleşmedir. Bu modlar hakkında detaylı bilgi almak için <https://fastbitlab.com/spi-bus-configuration-discussion-full-duplex-half-duplex-simplex/> linkteki yazıyı okuyabiliriz.

Master

## Konfigürasyon Kısımları



| Pin | Signal on Pin | GPIO o... | GPIO m...   | GPIO ...  | Maximu...  | User ... | Modif...                 |
|-----|---------------|-----------|-------------|-----------|------------|----------|--------------------------|
| PA5 | SPI1_SCK      | n/a       | Alternat... | No pul... | Very Hi... |          | <input type="checkbox"/> |
| PA7 | SPI1_MOSI     | n/a       | Alternat... | No pul... | Very Hi... |          | <input type="checkbox"/> |

Mode  ▼

Hardware NSS Signal  Disable

## Basic Parameters

|              |           |
|--------------|-----------|
| Frame Format | Motorola  |
| Data Size    | 8 Bits    |
| First Bit    | MSB First |

## Clock Parameters

|                           |             |
|---------------------------|-------------|
| Prescaler (for Baud Rate) | 2           |
| Baud Rate                 | 8.0 MBits/s |
| Clock Polarity (CPOL)     | Low         |
| Clock Phase (CPHA)        | 1 Edge      |

### Advanced Parameters

|                 |          |
|-----------------|----------|
| CRC Calculation | Disabled |
| NSS Signal Type | Software |

## Kod Kismi

```
57 /* USER CODE BEGIN 0 */  
58 uint8_t TxBuffer[1] = {0};  
59 /* USER CODE END 0 */  
  
90 /* USER CODE BEGIN 2 */  
91 int i=0;  
92 /* USER CODE END 2 */
```

```

96     while (1)
97     {
98         /* USER CODE END WHILE */
99
100        /* USER CODE BEGIN 3 */
101        for(i= 0; i< 50; i++){
102            TxBuffer[0] = i;
103
104            HAL_SPI_Transmit (&hspi1, TxBuffer, sizeof(TxBuffer), 1000);
105            HAL_Delay(1000);
106        }
107    }
108    /* USER CODE END 3 */
109 }

```

## Slave

### Konfigürasyon Kısımları

| Pin ... | Signal on Pin | GPIO ou... | GPIO m...   | GPIO P...    | Maximu... | User La... | Modifi...                |
|---------|---------------|------------|-------------|--------------|-----------|------------|--------------------------|
| PD12    | n/a           | Low        | Output P... | No pull-u... | Low       |            | <input type="checkbox"/> |
| Pin ... | Signal on Pin | GPIO ou... | GPIO m...   | GPIO P...    | Maximu... | User La... | Modifi...                |
| PA5     | SPI1_SCK      | n/a        | Alternat... | No pull-u... | Very High |            | <input type="checkbox"/> |
| PA7     | SPI1_MOSI     | n/a        | Alternat... | No pull-u... | Very High |            | <input type="checkbox"/> |

Mode: Receive Only Slave

Hardware NSS Signal: Disable

#### Basic Parameters

|              |           |
|--------------|-----------|
| Frame Format | Motorola  |
| Data Size    | 8 Bits    |
| First Bit    | MSB First |

#### Clock Parameters

|                       |        |
|-----------------------|--------|
| Clock Polarity (CPOL) | Low    |
| Clock Phase (CPHA)    | 1 Edge |

#### Advanced Parameters

|                 |          |
|-----------------|----------|
| CRC Calculation | Disabled |
| NSS Signal Type | Software |

## Kod Kısımları

```

44 /* USER CODE BEGIN PV */
45 uint8_t RxBuffer[1];
46 /* USER CODE END PV */

```

```
96     while (1)
97     {
98         /* USER CODE END WHILE */
99
100        /* USER CODE BEGIN 3 */
101        HAL_SPI_Receive (&hspi1, RxBuffer, sizeof(RxBuffer), 1000);
102
103        if (RxBuffer[0] == 3)
104        {
105            HAL_GPIO_WritePin(GPIOD, GPIO_PIN_12, GPIO_PIN_SET);
106        }
107        else{
108            HAL_GPIO_WritePin(GPIOD, GPIO_PIN_12, GPIO_PIN_RESET);
109        }
110        HAL_Delay(1000);
111    }
112    /* USER CODE END 3 */
113 }
```