《電子技術應用》
您所在的位置:首頁 > 嵌入式技術 > 解決方案 > STM8 MCKIT1.0 BLDC無感控制中的采集實現分析

STM8 MCKIT1.0 BLDC無感控制中的采集實現分析

2016-05-19
關鍵詞: BLDC

       ST寫的STM8電機庫中,可以用無感方式驅動BLDC。STM8芯片只有一個AD轉換器,而且是8位機。庫中要對反

電勢采集,比較處理,還要進行母線電壓,母線電流、散熱片溫度、電位器等模擬信號進行采集、運算和處理。

這優先級處理說起來是很重要的。不過,ST把程序寫的很好,至少我這樣認為。它把不同的任務放到準確的時間段里進行采集處理。
先把它核心AD采集處理帖出來再分析 。

#ifdef SENSORLESS 
 @near @interrupt @svlreg void ADC2_IRQHandler (void)
 {
  if (ADC_State == ADC_SYNC)
  {
   // Syncronous sampling
   
   u16 data;
   u8 delay;
   u16 bemf_threshold;

   // Reset bit
   bComHanderEnable = 0;
    
   //clear interrupt flag
   ADC2->CSR &= (u8)(~BIT7);
     
   //left align - read DRH first
   data = ADC2->DRH;
   data <<= 2;
   data |= (ADC2->DRL & 0x03);  
   
   switch( ADC_Sync_State )
   {
    case ADC_BEMF_INIT:
     ADC2->CSR = (u8)((Current_BEMF_Channel|BIT5));
     BEMF_Sample_Debounce = 0;
     Zero_Sample_Count = 0;
     ADC_Sync_State = ADC_BEMF_SAMPLE;
     SetSamplingPoint_BEMF();
    break;

    case ADC_BEMF_SAMPLE:
     //detect zero crossing
     if( Current_BEMF == BEMF_FALLING )
     {
      if( Z_Detection_Type == Z_DETECT_PWM_OFF )
      {
       bemf_threshold = BEMF_FALLING_THRESHOLD;
      }
      else
      {
       bemf_threshold = hNeutralPoint;
      }

      if (Ramp_Step > FORCED_STATUP_STEPS)
      {
       if( data <  bemf_threshold  )
       {
        Zero_Sample_Count++;
        BEMF_Sample_Debounce++;
        if( BEMF_Sample_Debounce >= BEMF_SAMPLE_COUNT )
        {
         hTim3Th -= hTim3Cnt;
         GetStepTime();
 
         SpeedMeasurement();

         bComHanderEnable = 1;

         BEMF_Sample_Debounce = 0;
        }
       }
       else
       {
        BEMF_Sample_Debounce = 0;
       }
      }
     }
     else
     {
      if( Z_Detection_Type == Z_DETECT_PWM_OFF )
      {
       bemf_threshold = BEMF_RISING_THRESHOLD;
      }
      else
      {
       bemf_threshold = hNeutralPoint;
      }
  
      if (Ramp_Step > FORCED_STATUP_STEPS)
      {
       if( data > bemf_threshold )
       {
        Zero_Sample_Count++;
        BEMF_Sample_Debounce++;
        if( BEMF_Sample_Debounce >= BEMF_SAMPLE_COUNT )
        {
         hTim3Th -= hTim3Cnt;
         GetStepTime();
  
         SpeedMeasurement();

         bComHanderEnable = 1;

         BEMF_Sample_Debounce = 0;
        }
       }
       else
       {
        BEMF_Sample_Debounce = 0;
       }
      }
     }
    break;

    case ADC_CURRENT_INIT:
     ADC2->CSR = (ADC_CURRENT_CHANNEL|BIT5);
     ADC_Sync_State = ADC_CURRENT_SAMPLE;
     SetSamplingPoint_Current();
    break;

    default:
    case ADC_AVCURRENT_INIT:
     ADC2->CSR = (ADC_AVCURRENT_CHANNEL|BIT5);
     ADC_Sync_State = ADC_AVCURRENT_CHANNEL;// ADC_USER_SYNC_SAMPLE;
     SetSamplingPoint_AVCURRENT();
    break;

  
    case ADC_CURRENT_SAMPLE:
     ADC_Buffer[ ADC_CURRENT_INDEX ] = data;
     break;

    case ADC_AVCURRENT_SAMPLE:
     ADC_Buffer[ ADC_AVCURRENT_INDEX] = data;
     break;
   }

   // Store the current channel selected
   bCSR_Tmp = ADC2->CSR;

   // Set the Async sampling channel
   switch (ADC_Async_State)
   {
    default:
    case ADC_BUS_INIT:
     ADC2->CSR = (ADC_BUS_CHANNEL|BIT5);
     ADC_Async_State = ADC_BUS_SAMPLE;
    break;
    
    case ADC_TEMP_INIT:
     ADC2->CSR = (ADC_TEMP_CHANNEL|BIT5);
     ADC_Async_State = ADC_TEMP_SAMPLE;
    break;
   
    case ADC_USER_ASYNC_INIT:
     ADC2->CSR = (ADC_USER_ASYNC_CHANNEL|BIT5);
     ADC_Async_State = ADC_USER_ASYNC_SAMPLE;
    break;
   }

    // Disable ext. trigger
    ADC2->CR2 &= (u8)(~BIT6);
    //Start ADC sample
    ADC2->CR1 |= BIT0;


   ADC_State = ADC_ASYNC;
   
   if (bComHanderEnable == 1)
   {
    ComHandler();
   }
  }
  else
  {
   // Syncronous sampling
   u16 data;
   
   data = ADC2->DRH;
   data <<= 2;
   data |= (ADC2->DRL & 0x03);

   //clear interrupt flag
   ADC2->CSR &= (u8)(~BIT7);

   // Restore the sync ADC channel
   ADC2->CSR = bCSR_Tmp;
 

    // Enable ext. trigger
    ADC2->CR2 |= BIT6;


   // Manage async sampling
   switch (ADC_Async_State)
   {
    default:
    case ADC_BUS_SAMPLE:
     ADC_Buffer[ ADC_BUS_INDEX ] = data;
     ADC_Async_State = ADC_TEMP_INIT;
    break;

    case ADC_TEMP_SAMPLE:
     ADC_Buffer[ ADC_TEMP_INDEX ] = data;
     ADC_Async_State = ADC_USER_ASYNC_INIT;
    break;

    case ADC_USER_ASYNC_SAMPLE:
     ADC_Buffer[ ADC_USER_ASYNC_INDEX ] = data;
     ADC_Async_State = ADC_BUS_INIT;
    break;
   }
   
   ADC_State = ADC_SYNC;   
  }
 }
#endif

 

上面的代碼我改了一點點,就是多采集了一路平均電流。

AD采集分兩種,一個是同步,一個是異步。同步中有三個采集通道,異步中有三個采集通道。同步中的通道為反電勢通道、瞬時電流、平均電流。異步采集中通道為母線電壓、溫度值、電位器。

異步采集是在同步完成后進行的。同步采集是通過TIM1的通道4觸發采集。

所以每個PWM周期采集2路模擬信號。異步采集的通道與PWM的ON與OFF狀態無關,所以安排在異步采集中。同步采集中的反電勢需要在PWM固定 時刻采集,或ON或OFF,看BEMF的過零比較方案。瞬時電流一般在TON時刻采集。因為原來ST有PWM特殊時刻做了一路用戶通道中,所以我就把平均 電流加到這一通道上了。其實平均電流采集也可以放到異步中。無所謂了,功能實現是沒問題的。

另外,異步采集中的反電勢通道一直是設為浮空相的通道的。而且反電勢的采集在D與Z之間,即退磁結束與過零點之間進行的異步采集均為反電勢,而瞬時 電流的采集是在Z與C之間,即過零與換相之間進行的異步采集均為瞬時瞬時電流。所以用戶的通道(平均電流)就是在換相與退磁之間了。

ST的無感方案,啟動方案感覺只能針對工業用電機,像在4極對下4K轉速的電機,那啟動參數不用怎么改。但如果改為航模電機,無論啟動 PWM改為多少,總是不能啟動成功。可能是我還找到巧門,也可能沒設對參數,對于高速電機,像這種無感啟動可能是升頻升壓法啟動才可靠。我早期寫的例程, 無論什么電機,用的是升頻升壓法,無論什么電機,都可以正常啟動,只是啟動過程(大約1S)電流從大到小,,至少正常運行至最小電流值。

本站內容除特別聲明的原創文章之外,轉載內容只為傳遞更多信息,并不代表本網站贊同其觀點。轉載的所有的文章、圖片、音/視頻文件等資料的版權歸版權所有權人所有。本站采用的非本站原創文章及圖片等內容無法一一聯系確認版權者。如涉及作品內容、版權和其它問題,請及時通過電子郵件或電話通知我們,以便迅速采取適當措施,避免給雙方造成不必要的經濟損失。聯系電話:010-82306118;郵箱:aet@chinaaet.com。
主站蜘蛛池模板: 国内精品91最新在线观看 | 国产大秀视频 | 欧美手机看片 | 免费看欧美一级特黄a毛片 免费看片aⅴ免费大片 | 国产高清三级 | 久久99国产亚洲精品观看 | 久久爱一区| 欧美精品三区 | 久久久国产一区二区三区丝袜 | 亚洲性影院| 99pao在线视频成精品 | 男女晚上爱爱的视频在线观看 | 国产自产在线 | 韩国主播19福利视频在线 | 欧美一区二区在线观看免费网站 | 欧美日韩免费一区二区在线观看 | 亚洲黄色成人 | 亚洲成人综合视频 | 久久久久久综合一区中文字幕 | 国产精品久久国产精品99 | 一级毛片真人不卡免费播 | 欧美精品束缚一区二区三区 | 国产成人精品男人免费 | 精品国产区一区二区三区在线观看 | 免费福利入口在线观看 | 日本一在线中文字幕天堂 | 一区二区三区国产美女在线播放 | 国产高清毛片 | 8050网午夜一级毛片免费不卡 | 国产亚洲男人的天堂在线观看 | 国产亚洲图片 | 欧美午夜不卡在线观看最新 | 在线视频一区二区三区在线播放 | 亚洲免费区 | 中文字幕一区二区小泽玛利亚 | 亚洲制服丝袜美腿亚洲一区 | 久久99国产亚洲精品 | 中文字幕亚洲综合久久 | 草草影院ccyy国产日本欧美 | 在线观看人成网站深夜免费 | 欧美日韩一区二区三区在线观看 |