Отличие этого проекта от представленного ранее(на ATmega8) состоит в том, что в данной схеме применен микроконтроллер ATmega48, который тактируется от внешнего кварца частотой 16МГц, тем самым была увеличена частота ШИМ импульсов. Для регулировки частоты вращения двигателя используется потенциометр, подключенный к входу ADC3. Переменная speed
хранит значение АЦП. При вращении ручки потенциометра, если минимальный порог START_PWM
будет превышен, двигатель начнет вращение и можно увеличивать обороты, при обратном вращении ручки бесколлекторный двигатель будет снижать обороты и при снижении минимального порога START_PWM
двигатель остановится.
Схема устройства
Исходный текст программы:
// Управление бесколлекторным двигателем постоянного тока без датчиков на ATmega48 #include <avr/io.h> #include <avr/interrupt.h> #include <util/delay.h> #include <avr/wdt.h> // Фаза U(Верхнее плечо) #define UH_ON TCCR2A |= (1 << COM2A1); #define UH_OFF TCCR2A &= ~(1 << COM2A1); // Фаза U(Нижнее плечо) #define UL_ON PORTB |= (1 << PB5); #define UL_OFF PORTB &= ~(1 << PB5); // Фаза V(Верхнее плечо) #define VH_ON TCCR1A |= (1 << COM1B1); #define VH_OFF TCCR1A &= ~(1 << COM1B1); // Фаза V(Нижнее плечо) #define VL_ON PORTB |= (1 << PB4); #define VL_OFF PORTB &= ~(1 << PB4); // Фаза W(Верхнее плечо) #define WH_ON TCCR1A |= (1 << COM1A1); #define WH_OFF TCCR1A &= ~(1 << COM1A1); // Фаза W(Нижнее плечо) #define WL_ON PORTB |= (1 << PB0); #define WL_OFF PORTB &= ~(1 << PB0); #define PHASE_ALL_OFF UH_OFF;UL_OFF;VH_OFF;VL_OFF;WH_OFF;WL_OFF; #define SENSE_U ADMUX = 0; // Вход обратной ЭДС фазы U #define SENSE_V ADMUX = 1; // Вход обратной ЭДС фазы V #define SENSE_W ADMUX = 2; // Вход обратной ЭДС фазы W #define SENSE_UVW (ACSR&(1 << ACO)) // Выход компаратора #define START_PWM 50 // Минимальный ШИМ при запуске unsigned char position, speed; volatile unsigned char commutation_step = 0; volatile unsigned char rotor_run = 0; // Счетчик импульсов обратной ЭДС // Крутим по часовой стрелке void commutation(char startup) { switch (commutation_step) { case (0): if(!SENSE_UVW || startup) { UH_ON; // На фазе U - ШИМ WH_OFF; // Фаза W отключена SENSE_W; // Активируем вход фазы W commutation_step = 1; // Следующий шаг TCNT0 = 0; // Обнуляем счетчик T1 } break; case (1): if(SENSE_UVW || startup) { VL_OFF; // На фазе V - лог. 0 WL_ON; // На Фазе W - лог. 1 SENSE_V; // Активируем вход фазы V commutation_step = 2; TCNT0 = 0; // Обнуляем счетчик T1 } break; case (2): if(!SENSE_UVW || startup) { UH_OFF; // Фаза U отключена VH_ON; // На фазе V - ШИМ SENSE_U; // Активируем вход фазы U commutation_step = 3; TCNT0 = 0; // Обнуляем счетчик T1 } break; case (3): if(SENSE_UVW || startup) { UL_ON; // На фазе U - лог. 1 WL_OFF; // На Фаза W - лог. 0 SENSE_W; // Активируем вход фазы W commutation_step = 4; TCNT0 = 0; // Обнуляем счетчик T1 } break; case (4): if(!SENSE_UVW || startup) { VH_OFF; // Фаза V отключена WH_ON; // На фазе W - ШИМ SENSE_V; // Активируем вход фазы V commutation_step = 5; TCNT0 = 0; // Обнуляем счетчик T1 } break; case (5): if(SENSE_UVW || startup) { UL_OFF; // На фазе U - лог. 0 VL_ON; // На Фазе V - лог. 1 SENSE_U; // Активируем вход фазы U commutation_step = 0; TCNT0 = 0; // Обнуляем счетчик T1 } break; } } // Обработчик прерывания по компаратору. Детектор обратной ЭДС ISR(ANALOG_COMP_vect) { if(rotor_run == 200) commutation(0); rotor_run++; if(rotor_run > 200) { rotor_run = 200; wdt_reset(); } } // Обработчик прерывания по переполнению Т0. Работа двигателя без сигналов обратной ЭДС ISR(TIMER0_OVF_vect) { commutation(1); // Если сработало прерывание, есть пропуски импульсов обратной ЭДС rotor_run = 0; // Сбрасываем счетчик импульсов OCR1A = START_PWM; // ШИМ минимум OCR1B = START_PWM; OCR2A = START_PWM; } int main( void ) { //Watchdog on wdt_enable(WDTO_1S); // Порты ввода/вывода DDRB = 0xFF; // Порт B - выход PORTB = 0x00; // T0 - для старта и работы двигателя без сигналов обратной ЭДС TCCR0A = 0; TCCR0B = 0; TCCR0B |= (1 << CS02)|(1 << CS00); // Предделитель на 1024 TIMSK0 |= (1 << TOIE0); // Разрешаем прерывание по переполнению T0 // T1 и T2 ШИМ TCCR1A |= (1 << WGM10); // Режим Fast PWM, 8-bit TCCR1B |= (1 << CS10)|(1 << WGM12); // Без предделителя TCCR2A |= (1 << COM2A1)| // Сброс вывода OC2A при совпадении (1 << WGM21)|(1 << WGM20); // Режим Fast PWM TCCR2B |= (1 << CS20); // Без предделителя PHASE_ALL_OFF; // Выключаем все фазы ADCSRB |= (1 << ACME); // Отрицательный вход компаратора подключаем к выходу мультиплексора АЦП DIDR1 |= (1 << AIN0D); sei(); // Глобально разрешаем прерывания while(1) { cli(); position = ADMUX; // Сохраняем позицию ротора в буфер ADMUX = (1 << MUX1)|(1 << MUX0); // Вход ADC3 ADCSRA |= (1 << ADEN) | (1 << ADPS1)|(1 << ADPS0); // Разрешаем АЦП, предделитель на 8 ADCSRA |= (1 << ADSC); // Запускаем преобразование АЦП while(ADCSRA & (1 << ADSC)){}; // Ждем окончания преобразования speed = ADC/4; ADCSRA = 0; // Выключаем АЦП ADCSRB = 0; ADCSRB |= (1 << ACME); // Отрицательный вход компаратора подключаем к выходу мультиплексора АЦП ADMUX = position; sei(); if(speed > START_PWM) { ACSR |= (1 << ACIE); // Разрешаем прерывание от компаратора TIMSK0 |= (1 << TOIE0); // Разрешаем прерывание по переполнению T0 if(rotor_run == 200) { // Если импульсы обратной ЭДС присутствуют, крутим наполную OCR1A = speed; OCR1B = speed; // Изменяем ШИМ OCR2A = speed; } } else { PHASE_ALL_OFF; // Все фазы выключены ACSR &= ~(1 << ACIE); // Запрещаем прерывание от компаратора TIMSK0 &= ~(1 << TOIE0); // Запрещаем прерывание по переполнению T0 } } }
Видео работы устройства
Архив для статьи "Управление бесколлекторным двигателем постоянного тока без датчиков на ATmega48" | |
Описание:
Проект AVRStudio4, макет печатной платы DipTrace |
|
Размер файла: 54.28 KB Количество загрузок: 1 062 | Скачать |
Комментарии
Здравствуйте, все данные в разделе "Контакты"
Здравствуйте.
HIGH FUSE: 0xDF
LOW FUSE: 0xEF