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

Спящий режим МК AVR


Рекомендуемые сообщения

Сегодня пошаманю со спящими режимами. Возможно, это глюк протеуса.

Цитата

if (!(PIND&0x4)) //обработка нажатия основной кнопки

Лучше использовать сдвиг и именованную константу, иначе можно подумать что кнопка висит на PD4, а не на PD2:
 

#define BTN 2

if(! (PIND & (1<<BTN)) )
Цитата

FRONT_LAMP_MODE++; //увеличиваем значение переменной режима

В Си принято капсом обозначать константы, а не переменные.

Цитата

PORTB=0x00; //выключаем все светодиоды
                        STROB_MODE=0; //отключаем режим стробоскопа
                        OCR1A=128; //переводим яркость переднего фонаря в режим "БЛИЖНИЙ СВЕТ"
                        FRONT_LAMP_MODE=0; //устанавливаем нулевой режим
                        sleep_cpu(); //отправляем МК в СОН
                        break;

не увидел запрещения-разрешения прерывания INT0. Без этого оно будет вызываться постоянно пока кнопка нажата.

Цитата

sei(); //глобально разрешаем прерывания
    counter_init(); //инициализируем счетчик    
    port_init(); //инициализируем порты
    PWM_timer_init(); //инициализируем ШИМ
    int_init(); //инициализируем прерырвания
    sleep_init(); //инициализируем режим сна
    sleep_cpu(); //засыпаем

разрешать прерывания стоит после настроек а не до них. Еще неплохо бы явным образом сбросить флаги прерываний.

Ругался на отсутствие форматирования исходного кода (включая отсутствие осмысленных комментариев и наличие неубранного после конфигуратора мусора) не менее 15 раз.

Часть моих наработок.

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

Реклама: ООО ТД Промэлектроника, ИНН: 6659197470, Тел: 8 (800) 1000-321

55 минут назад, COKPOWEHEU сказал:

Сегодня пошаманю со спящими режимами. Возможно, это глюк протеуса.

Было бы здорово! Заранее спасибо!

55 минут назад, COKPOWEHEU сказал:

не увидел запрещения-разрешения прерывания INT0. Без этого оно будет вызываться постоянно пока кнопка нажата.

Поправил, правда это делается в регистре GICR, то есть

GICR |= (1<<INT0) //разрешаем прерывания

GICR &= ~(1<<INT0) //запрещаем прерывания

При включении прерываний, по умолчанию включено прерывание именно по низкому уровню на INT0, остальные виды прерываний уже настраиваются в регистре  MCUCR.

Так же учту все остальные поправки. Спасибо!

55 минут назад, COKPOWEHEU сказал:

не увидел запрещения-разрешения прерывания INT0. Без этого оно будет вызываться постоянно пока кнопка нажата.

Правда всё равно немного не понимаю, выглядит это всё вот так:

GICR |= (1<<INT0) //разрешаем прерывания
sleep_cpu(); //отправляем МК в сон
GICR &= ~(1<<INT0) //запрещаем прерывания

а как же же МК должен проснуться, если прерывания мы запретили?...Вот сейчас проэксперементировал и сначала запретил прерывания, а потом разрешил, после чего МК у меня успешно заснул. Но теперь не хочет просыпаться, сколько кнопку не тыкай...

GICR &= ~(1<<INT0) //запрещаем прерывания
sleep_cpu(); //отправляем МК в сон
GICR |= (1<<INT0) //разрешаем прерывания

 

Изменено пользователем -=FISHER=-

Мы все учились по-немногу, чему-нибудь и как-нибудь...

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

20% скидка на весь каталог электронных компонентов в ТМ Электроникс!

Акция "Лето ближе - цены ниже", успей сделать выгодные покупки!

Плюс весь апрель действует скидка 10% по промокоду APREL24 + 15% кэшбэк и бесплатная доставка!

Перейти на страницу акции

Реклама: ООО ТМ ЭЛЕКТРОНИКС, ИНН: 7806548420, info@tmelectronics.ru, +7(812)4094849

Заметил интересную особенность, в Протеусе у меня МК просыпается даже если я вообще никак не инициализирую прерывания!

То есть даже если закомментировать эти строчки. МК всё равно просыпается при появлении "ЛОГ.0" на PD2.

//GICR |=(1<<INT0);
//sei();

Теоретически это ведь не возможно?

А я понял почему! У меня в процедуре инициализации таймера-счетчика есть вот такая строчка SREG = (0x80), которая устанавливает в единичку бит I, регистра SREG, который в свою очередь и включает глобальные прерывания.

Изменено пользователем -=FISHER=-

Мы все учились по-немногу, чему-нибудь и как-нибудь...

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

Выбираем схему BMS для корректной работы литий-железофосфатных (LiFePO4) аккумуляторов

 Обязательным условием долгой и стабильной работы Li-FePO4-аккумуляторов, в том числе и производства EVE Energy, является применение специализированных BMS-микросхем. Литий-железофосфатные АКБ отличаются такими характеристиками, как высокая многократность циклов заряда-разряда, безопасность, возможность быстрой зарядки, устойчивость к буферному режиму работы и приемлемая стоимость. Но для этих АКБ, также как и для других, очень важен контроль процесса заряда и разряда, а специализированных микросхем для этого вида аккумуляторов не так много. Инженеры КОМПЭЛ подготовили список имеющихся микросхем и возможных решений от разных производителей. Подробнее>>

Реклама: АО КОМПЭЛ, ИНН: 7713005406, ОГРН: 1027700032161

Проверил. Вот этот набросок работает нормально. Единственное что по низкому уровню протеус генерирует всего одно прерывание, хотя должен постоянно. В моем коде используется файл pinmacro.h, это макросы для удобной работы с портами. Он есть по ссылке в моей подписи, но и без того несложно понять.

Прерывание разрешаем только перед уходом в сон. Иначе оно будет отъедать 90% скорости на постоянный вызов пустого обработчика. А после выхода из сна - запрещаем. Грубо говоря, не акцентируясь на конкретном прерывании, так

	sei(); //разрешаем прерывание только когда оно нужно.
	sleep_cpu(); //эта команда не завершится пока контроллер не проснется
	cli(); //и уже когда он проснулся, снова запрещаем прерывание
	

sleep_my.c

Ругался на отсутствие форматирования исходного кода (включая отсутствие осмысленных комментариев и наличие неубранного после конфигуратора мусора) не менее 15 раз.

Часть моих наработок.

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

7 часов назад, -=FISHER=- сказал:

Сразу после выхода из сна убеждаемся что кнопка ТОЧНО нажата а не была короткая помеха, иначе снова уходим в сон(цикл постоянного входа в сон, пока не будет установлен фак

Подскажите пожалуйста, как лучше и проще это сделать в моем случае?

Мы все учились по-немногу, чему-нибудь и как-нибудь...

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

Цикл на 10 итераций, внутри задержка на 2..4мс и считывание кнопки - если минимум 5 раз подряд кнопка нажата, то засчитать нажатие иначе - переходим к команде входа в спящий режим.

Учение - изучение правил. Опыт - изучение исключений.

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

1 час назад, Alexeyslav сказал:

Цикл на 10 итераций, внутри задержка на 2..4мс и считывание кнопки

Цикл for как я понимаю?

Изменено пользователем -=FISHER=-

Мы все учились по-немногу, чему-нибудь и как-нибудь...

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

Можешь и for, в языке множество способов организовывать циклы, вплоть до разворачивания в линейный код...

Учение - изучение правил. Опыт - изучение исключений.

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

11 час назад, Alexeyslav сказал:

Можешь и for

Этот цикл лучше записать вначале main не трогая весь остальной код что у меня есть уже?

Мы все учились по-немногу, чему-нибудь и как-нибудь...

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

А смысл тогда? Проверка нужна везде где ждёшь нажатия а не только на старте...

Учение - изучение правил. Опыт - изучение исключений.

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

2 минуты назад, Alexeyslav сказал:

А смысл тогда?

Плоховато понял, не могли бы Вы хотя бы в общих чертах набросать эту мысль в коде?

Изменено пользователем -=FISHER=-

Мы все учились по-немногу, чему-нибудь и как-нибудь...

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

Если у тебя проверка на нажатие только в начале цикла, то туда и ставь. А смысл такой проверки - чтобы отсеять ложные срабатывания и не реагировать на дребезг. Иначе, контроллер же очень быстрый - воспримет дребезг как сотню нажатий и за это время 100 раз успеет поспать и что-то сделать согласно алгоритму. В принципе, может хватить лишь одной проверки спустя 20-40 мс после первой регистрации нажатия(срабатывание прерывания и выход из сна), но к хорошему надо привыкать и стараться делать как надо - потом этот алгоритм ещё в какой проект утащишь и т.д.

Учение - изучение правил. Опыт - изучение исключений.

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

11 час назад, Alexeyslav сказал:

 но к хорошему надо привыкать и стараться делать как надо - потом этот алгоритм ещё в какой проект утащишь и т.д.

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

Мы все учились по-немногу, чему-нибудь и как-нибудь...

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

  • 1 месяц спустя...

Здравствуйте! Пишу программу для Atmega8a на Atmel Studio 7, эмулирую на Proteus 8. Решил ввести режим СНА так называемый  POWER DOWN для остановки тактового генератора (внутреннего или внешнего- без разницы).  Накидал строки для подготовки, сна и выхода из сна... Но! некорректно работает выход из сна. Родной атмеловский сайт и прочие мурзилки говорят, что при выходе из сна МК должен ПРОДОЛЖИТЬ работу с места спячки, а у меня чего-то поучается сон-и сразу после выхода из сна - в начало заголовка main по коду. Да, еще, выход из сна обеспечивается по INT0- так мне нужно.

Код ниже:

***************

// ATmega8a
#define F_CPU 1000000UL // частота МК 1МГц

#include <avr/interrupt.h>
#include <avr/sleep.h> // здесь описаны режимы сна
#include <avr/io.h>  //  Подключили описание контроллера
#include <util/delay.h> // Подключили библиотеку задержек

int main(void)
{
    DDRD = 0b00000000;
    DDRC = 0b00110000; 
    PORTC |= (1 << PC4); 
    _delay_ms(1000);
    PORTC &= ~(1 << PC4);
    _delay_ms(1000);
    PORTC |= (1 << PC4);

                                     // set sleep mode
        GICR|=(1<<6);               //разрешаем прерывание по INT0
        GIFR|=(1<<6);
        MCUCR|=((0<<1)|(0<<0));     //прерывание по 0 сигнала на INT0
        sei();                      // разрешаем прерывания глобальные
        
        set_sleep_mode(SLEEP_MODE_PWR_DOWN);
        cli();
    while(1)
    {
            if (PINC &= 1<<PC0) 
            {
                            // плюс внутри while планируется писат сой код....
                            // sleep_mode() has a possible race condition
        sleep_enable();
        sei();
        _delay_ms(1000);
        sleep_cpu();        // тут засыпаем и с этой же точки проснемся (по идее???   и дальше продолжить вниз по коду, зажечь диод на PC5)
    
        sleep_disable();
        GICR|=(0<<6);       // отключить прерывание по инт0
        GIFR|=(0<<6);
        
        PORTC |=(1<<PC5);     // идентификатор того что программа после сна выполняется дальше не перескакует в начало на main                     
            }
        sei(); 
      }
}

************************************************

Просто нужно именно, чтобы код продолжал выполняться дальше после сна, т.е.  не блымал светодиод на РС5, а не перескакивал на main в начало.

son.c

proteus.PNG

SLEPP_MODE.pdsprj

SLEPP_MODE.pdsprj.AdminPC.Admin.workspace

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

А обработчик прерывания вы написали? При возникновении прерывания идет прыжок именно в обработчик и только после него обратно в программу. Если прерывание не нужно, объявите его как EMPTY_INTERRUPT(INT0_vect); вместо обычного ISR();

Ругался на отсутствие форматирования исходного кода (включая отсутствие осмысленных комментариев и наличие неубранного после конфигуратора мусора) не менее 15 раз.

Часть моих наработок.

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

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

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

Гость
Unfortunately, your content contains terms that we do not allow. Please edit your content to remove the highlighted words below.
Ответить в этой теме...

×   Вставлено с форматированием.   Восстановить форматирование

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

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

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

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

Загрузка...
  • Последние посетители   0 пользователей онлайн

    • Ни одного зарегистрированного пользователя не просматривает данную страницу
  • Сообщения

    • @Gomerchik а вы контролировали как меняется уровень сигнала на А1 ардуины?
    • Спасибо за совет. Автором данного проекта я не являюсь, мне нужно было воссоздать уличный датчик для метеостанции взамен пропавшего(( Из разного найденного в интернете этот проект работает с моей станцией Орегон (спасибо автору). В понедельник попробую последовать Вашему совету. Но все равно куча непоняток  как блин это работает)) Если дело в неправильной отправки команды, то как на это влияет подключение датчика температуры? Если совсем не подключать таймер, то передача идет один раз (как и прописано в программе), станция принимает и отображает, но минут через сколько-то естественно станция уже ни чего не показывает, но с таймером питание полностью не пропадает с ардуинки, но передача сигнала каким-то образом работает по таймеру.  В моем понимании данная команда подается один раз потому, что таймер должен отключать питание МК после передачи сигнала и каждые 43 сек снова подавать питание (так того требует станция).  Ардуино передает показания температуры отключается полностью и 43 секунды мк не работает.  Сейчас у меня питание пока сделано на подпитке от солнечной батареи, но пару пасмурных дней и аккумулятор съедается до отключения(
    • thickman Так и сделаю. Вытащу из бу БП.  Буду знать, как отличить. Благодарю. Заменил транзисторы на IRFB20N50K. Картина стала, совсем другой.  Похоже трудность не в драйвере, на момент подвозбуда, переходные процессы, в нем, завершены. Увеличил затворные резисторы до 50ом, стало немного лучше.  Не понятно, почему верхний ключ греется несколько сильнее. Возможно, стоит посмотреть ток в коллекторе.  Снабберные емкости временно удалил, изменений не произошло.  Замена ТГР на другой, на кольце MSTN-16A-TH, так же, результата не принесла.   irfb20n50k.pdf
    • А что нить из ассортимента активных щупов производства СССР..))
    • Типа такого: https://aliexpress.ru/item/2044864227.html?sku_id=58855020183
    • поняли неправильно. У ТЛ494 никакой защиты нет, усилители ошибки не защита, они не должны приводить к ложным импульсам. Причем тут "микруха" ?  надо нагружать ВСЁ. До сих пор вообще непонятно о каком ИИП идет речь и сколько у него каналов. Бесполезно схему рисовать? - Помогать так бесполезно. Картина кривая по самое "немогу" продолжаются картинки, снятые через ногу и без цены деления.
×
×
  • Создать...