IAR ile STM8S – PWM

Bir önceki yazıda Timer konusunu tamamladığımıza göre bu bölümde PWM’e geçebiliriz. Burada Timer’dan çok farklı bir şey olmayacak sadece Timer’ları ilgili çıkışlara yönlendireceğiz. STM8S de bulunan bir güzel özellik Timer’ın çıkışı almaktan ziyade bu çıkışlarını “Complement”‘ini almak da mümkündür. Bahsettiğim bu yapıyı genellikle 32-Bit MCU’larda görüyordum. 8 bit bir MCU’da bu özelliği görmez oldukça güzel. Bu sayede Half Bridge – Full Bridge uygulamalarını rahatlıkla gerçekleştirebileceğiz. Kafanızdaki soruya hemen cevap vereyim. Evet “Dead Time” özelliği bile var. 

 

 

Öncelikle hangi Timer ve çıkışları kullanacağımızı belirleyelim. Burada sadece TIM1 ve TIM2 den çıkış alabileceğimizi unutmuyoruz. Örnek olarak TIM1’in 1 ve 2 çıkışlarını ele alalım.


Buradan görebileceğiniz üzere 1 ve 2 nolu çıkışlar C6, C7 pinlerinde bulunuyor. Bunun yanı sıra bu özel “Alternatife Function” seçeneklerini kullanabilmek için Option Bytes kısmından ilgili AFRx ayarlarını “Alternative Function” olarak ayarlamanız gerekiyor. Bu işi IAR’da debug moduna geçiş (RUN etmeden önce) ST-Link->Option Bytes kısmından yapabilirsiniz.


Burada AFR7’yi de şimdiden aktif ederseniz daha sonraki kısımlarda kullanacağımız Complement çıkışlarını da açmış olursunuz.


Bu ayarları yaptıktan sonra “Save” butonuna basın ve projenin olduğu dizine kaydedin. Daha sonra bu dosyayı projeye dahil ederek her seferinde Option Bytes güncellemesini de otomatik olarak yapmış olursunuz.


Artık kod kısmını geçebiliriz. Öncelikle Timer ile frekans ayarını yapmalıyız.

TIM1_TimeBaseInit(Prescaler, TIM1_COUNTERMODE_UP, Period, 0);

Formül yine oldukça kolay.


Yapacağımız örnek 2kHz PWM için Period değerini 3999, Prescaler değerini ise 1 olarak ayarladım.

TIM1_TimeBaseInit(1, TIM1_COUNTERMODE_UP, TIM1_PERIOD, 0);

Bu işlemden sonra kullanacağınız PWM kanallarını aktif etmeniz gerekiyor. Bu işlemi yaparken başlangıç Duty değeri, Complement aktivasyonu, PWM modu gibi seçenekler var.

TIM1_OC1Init(TIM1_OCMODE_PWM2, TIM1_OUTPUTSTATE_ENABLE, TIM1_OUTPUTNSTATE_ENABLE,
1024, TIM1_OCPOLARITY_HIGH, TIM1_OCNPOLARITY_HIGH, TIM1_OCIDLESTATE_SET,
TIM1_OCNIDLESTATE_RESET); 

TIM1_OC2Init(TIM1_OCMODE_PWM2, TIM1_OUTPUTSTATE_ENABLE, TIM1_OUTPUTNSTATE_ENABLE,
1024, TIM1_OCPOLARITY_HIGH, TIM1_OCNPOLARITY_HIGH, TIM1_OCIDLESTATE_SET, 
TIM1_OCNIDLESTATE_RESET);

Burada PWM2 haricinde yapacağınız ayarlarda Toggle mode, Capture giriş gibi ekstra özelliklerde mevcut. OUTPUTNSTATE ile Complement çıkışı aktif ediyoruz.

Son olarak yapmamız gereken TIM1’i çalıştırıp PWM çıkışlarını aktif etmek

TIM1_Cmd(ENABLE);

TIM1_CtrlPWMOutputs(ENABLE);

Ben kütüphane içinde Duty değiştirmek için bir fonksiyon göremedim. İş başa düştü bunu da kendimiz yazarız artık.

static void TIM1_SetDuty(uint8_t oc, uint16_t period, uint8_t duty)
{
   uint16_t value = (uint16_t)((((uint32_t)period * (uint32_t)duty) / (uint32_t)(100)));
      
   switch(oc)
   {
      case 1: 
         TIM1->CCR1H = (uint8_t)(value >> 8);
         TIM1->CCR1L = (uint8_t)(value);
      break;
      
      case 2: 
         TIM1->CCR2H = (uint8_t)(value >> 8);
         TIM1->CCR2L = (uint8_t)(value);
      break;

      case 3:
         TIM1->CCR3H = (uint8_t)(value >> 8);
         TIM1->CCR3L = (uint8_t)(value);
      break;

      case 4: 
         TIM1->CCR4H = (uint8_t)(value >> 8);
         TIM1->CCR4L = (uint8_t)(value);
      break;
   }
}

Yüzdesel Duty hesabı için değişkenleri önce 32 bit’e “cast” ettim. Malum büyük period değerleri ve duty için çarpım işlemi 16 biti aşabilir. Bunun önlemini almamız gerekiyor. Artık kodu çalıştırıp çıkışları inceleyebiliriz.

#define TIM1_PERIOD (3999)

TIM1_SetDuty(1, TIM1_PERIOD, 25);
TIM1_SetDuty(2, TIM1_PERIOD, 50);

 


Tabi Dead Time’ı unutmamak gerekiyor. Yoksa yarım köprüleriniz hep yanar 🙂

TIM1_BDTRConfig(TIM1_OSSISTATE_ENABLE, TIM1_LOCKLEVEL_1, 117, TIM1_BREAK_ENABLE, TIM1_BREAKPOLARITY_HIGH, TIM1_AUTOMATICOUTPUT_ENABLE);

Çıkışları tekrar inceleyelim. Hatta biraz yakınlaştıralım.


Evet artık ölü zamanları ekleyerek verimli bir PWM elde ettik. Bu zamanı “117” değerini düzenleyerek değiştirebilirsiniz.

Kodların tam halini aşağıda paylaşıyorum.

#include "stm8s.h"

#define LED_GPIO        (GPIOB)
#define LED_PIN         ((GPIO_Pin_TypeDef)GPIO_PIN_5)

#define TIM1_PERIOD     (3999)

static void SYSTEM_Config(void);
static void CLK_Config(void);
static void GPIO_Config(void);
static void TIM1_Config(void);
static void TIM1_SetDuty(uint8_t oc, uint16_t period, uint8_t duty);


void main(void)
{
   SYSTEM_Config();
   
   TIM1_SetDuty(1, TIM1_PERIOD, 25);
   TIM1_SetDuty(2, TIM1_PERIOD, 50);
   
   for (;;)
   {

   }
}

static void SYSTEM_Config(void)
{
   CLK_Config();
   GPIO_Config();
   TIM1_Config();
}

static void CLK_Config(void)
{
   CLK_DeInit();
   CLK_HSECmd(DISABLE);
   CLK_HSICmd(ENABLE);
   CLK_ClockSwitchConfig(CLK_SWITCHMODE_AUTO, CLK_SOURCE_HSI, DISABLE, CLK_CURRENTCLOCKSTATE_DISABLE);
   CLK_SYSCLKConfig(CLK_PRESCALER_HSIDIV1);
   CLK_SYSCLKConfig(CLK_PRESCALER_CPUDIV1);
}

static void GPIO_Config(void)
{
   GPIO_Init(GPIOB, GPIO_PIN_5, GPIO_MODE_OUT_PP_LOW_FAST);
}

static void TIM1_Config(void)
{
   TIM1_DeInit();

   TIM1_TimeBaseInit(1, TIM1_COUNTERMODE_UP, TIM1_PERIOD, 0);

   TIM1_OC1Init(TIM1_OCMODE_PWM2, TIM1_OUTPUTSTATE_ENABLE, TIM1_OUTPUTNSTATE_ENABLE,
   1024, TIM1_OCPOLARITY_HIGH, TIM1_OCNPOLARITY_HIGH, TIM1_OCIDLESTATE_SET,
   TIM1_OCNIDLESTATE_RESET); 

   TIM1_OC2Init(TIM1_OCMODE_PWM2, TIM1_OUTPUTSTATE_ENABLE, TIM1_OUTPUTNSTATE_ENABLE,
   1024, TIM1_OCPOLARITY_HIGH, TIM1_OCNPOLARITY_HIGH, TIM1_OCIDLESTATE_SET, 
   TIM1_OCNIDLESTATE_RESET);
   
   TIM1_BDTRConfig(TIM1_OSSISTATE_ENABLE, TIM1_LOCKLEVEL_1, 117, TIM1_BREAK_ENABLE, TIM1_BREAKPOLARITY_HIGH, TIM1_AUTOMATICOUTPUT_ENABLE);
   
   TIM1_Cmd(ENABLE);

   TIM1_CtrlPWMOutputs(ENABLE);
}

static void TIM1_SetDuty(uint8_t oc, uint16_t period, uint8_t duty)
{
   uint16_t value = (uint16_t)((((uint32_t)period * (uint32_t)duty) / (uint32_t)(100)));
      
   switch(oc)
   {
      case 1: 
         TIM1->CCR1H = (uint8_t)(value >> 8);
         TIM1->CCR1L = (uint8_t)(value);
      break;
      
      case 2: 
         TIM1->CCR2H = (uint8_t)(value >> 8);
         TIM1->CCR2L = (uint8_t)(value);
      break;

      case 3:
         TIM1->CCR3H = (uint8_t)(value >> 8);
         TIM1->CCR3L = (uint8_t)(value);
      break;

      case 4: 
         TIM1->CCR4H = (uint8_t)(value >> 8);
         TIM1->CCR4L = (uint8_t)(value);
      break;
   }
}

#ifdef USE_FULL_ASSERT
void assert_failed(uint8_t* file, uint32_t line)
{ 
   while (1);
}
#endif

Toparlayacak Olursak;

Küçük ama dev sayılabilecek bu denetleyici ailesi PWM açısından da bize büyük katkılar sağlıyor. Gerek kolay çalıştırılabilir olması, gerek 32-Bit STM denetleyici aratmayacak kadar bol özelliğinin olması tasarımcılar açısından tercih sebebi olmasını sağlıyor.

Bir başka yazıda görüşmek üzere…

Esen Kalın.

You may also like...

Bir yanıt yazın

E-posta adresiniz yayınlanmayacak. Gerekli alanlar * ile işaretlenmişlerdir