Перейти к содержанию
    

Странное поведение МК при работающем АЦП

При работающем АЦП МК периодически сбрасывается, либо вообще наглухо зависает. Визуально это видно как сброс прибора в исходное состояние и работа сначала. Либо вообще зависает. Что самое интересное, сторожевой таймер включен. Как только выключаю измерения АЦП, прототип работает как часики. На семисегментном дисплее по кругу поочередно выводит состояние 3 переменных.

 

Блок питания трансформаторный. Не импульсный, чтобы грешить на выбросы напряжения. Входы АЦП: подключен один, остальные сначала висели в воздухе, затем просто капельками припоя посадил на минус.

 

Непонятное происходит. И проблема в том, что вообще непонятно куда рыть.

 

//========================================================================
//#pragma inline = forced
u16 read_adc (u08 channel)
{
   ADMUX = ((0<<REFS1) | (1<<REFS0) | (0<<ADLAR) | channel);

   set_bit (ADCSRA, ADSC);

   while (!(ADCSRA & (1<<ADIF)))
   {

   }

   set_bit (ADCSRA, ADIF);

   return ADC;
}
//========================================================================

//   adc = read_adc (PHASE_Usm); // Комментирую работу АЦП, МК нормально работает.

   adc = 512; // !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!

 

Цель прототипа измерение переменного тока. Цикл измерения длится 120 мс. Программа выстроена так, что зацикленность только при ожидании окончания преобразования ацп и окончании передачи данных по SPI. Цикл одного измерения длится 25 либо 13 тактов ацп. Это примерно от 60 до 30 мкс. То есть, сброс мк происходит явно из за зацикливания на ожидании окончания преобразования ацп. По сторожевому таймеру. Но иногда мк наглухо вешается. Менял ATMEGA32A на ATMEGA8535. Картина та же. Убираю измерение ацп, мк работает как часики.

Изменено пользователем demiurg1978

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Убираю измерение ацп, мк работает как часики.

Уберите зацикливание по биту готовности результата АЦП-преобразования.

Вместо него, после пуска АЦП, сделайте временнУю паузу, заведомо достаточную для завершения преобразования.

 

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Для улучшения точности преобразования рекомендуется "засыпать"

на время преобразования и просыпаться по прерыванию окончания преобразования.

Можно и не засыпать, но все равно работать по прерыванию.

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Вычислил причину. Срыв стека. В настройках IAR есть настройка задания стека. Самое смешное, что я сразу заподозрил это. Но стал грешить на срабатывание сторожевого таймера. Я сегодня игрался со скоростью преобразования АЦП, потом переделал на прерывания, после запуска преобразования перевод мк в спячку. При смене скорости МК сбрасывается. Но стало меняться время сбрасывания МК. Когда перевел на прерывания со спячкой, сброс МК прекратился, но показания периодически показывали какой сигнал на входе, хотя входы намертво на минус припаяны. Вот тут уже все стало показывать на срыв стека. Меняю параметры стека и усе заработало. Да едрить ево в качель! Три дня псу под хвост! Такие вот дела...

 

Проверил программу как на опросе флага, так и на прерываниях со спячкой. Как ни странно на последнем варианте светодиодный дисплей перестал мерцать. При опросе флага видно некоторое мерцание дисплея. В принципе, терпимо, но хотелось полностью исключить мерцание.

bool adc_complete;

void set_proc_meter_on (void)
{
  ACSR = (1<<ACD);

  ADCSRA = ((1<<ADEN) | (0<<ADSC) | (0<<ADATE) | (1<<ADIF) | (1<<ADIE) | (1<<ADPS2) | (0<<ADPS1) | (0<<ADPS0));

  _proc_meter = 1;
}

//========================================================================
//#pragma inline = forced
u16 read_adc (u08 channel) // Эта функция вызывается в обработчике прерывания по таймеру.
{
  ADMUX = ((0<<REFS1) | (0<<REFS0) | (0<<ADLAR) | channel);

  set_bit (ADCSRA, ADSC);

//   while (ADCSRA & (1<<ADSC))
//   {

//   }

//   set_bit (ADCSRA, ADIF);

  while (1)
  {
     __enable_interrupt ();

     sleep_mode_enable ();

     if (adc_complete)
     {
        adc_complete = false;
        break;
     }
  }

  return ADC;
}
//========================================================================

//========================================================================
#pragma vector = ADC_vect
__interrupt void adc_interrupt_handler (void)
{
  adc_complete = true;
}
//========================================================================

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Как ни странно на последнем варианте светодиодный дисплей перестал мерцать. При опросе флага видно некоторое мерцание дисплея. В принципе, терпимо, но хотелось полностью исключить мерцание.

LDM-208

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Присоединяйтесь к обсуждению

Вы можете написать сейчас и зарегистрироваться позже. Если у вас есть аккаунт, авторизуйтесь, чтобы опубликовать от имени своего аккаунта.

Гость
Ответить в этой теме...

×   Вставлено с форматированием.   Вставить как обычный текст

  Разрешено использовать не более 75 эмодзи.

×   Ваша ссылка была автоматически встроена.   Отображать как обычную ссылку

×   Ваш предыдущий контент был восстановлен.   Очистить редактор

×   Вы не можете вставлять изображения напрямую. Загружайте или вставляйте изображения по ссылке.

×
×
  • Создать...