MAL4X Научно-технический форум разработчиков симуляторов и автоматики


Симуляторы перегрузок. DIY электроника. ЭВМ. Компьютерные сети.
Up

ATMega328p и PowerDown

Техническая поддержка нахаляву, но не оперативная.

Модераторы: Death_Morozz, FELiS, null

ATMega328p и PowerDown

Сообщение thor_nsk » 01 ноя 2017, 21:42

Приветствую, уважаемые.
Надеюсь на вашу помощь в решении проблемы. Необходимо перевести мегу 328 в режим глубокого сна и выводить либо по нажатию кнопки, либо по ватчдогу раз в 8 секунд. Никак не хочет МК засыпать :(
Листинг программы и фьюзы прилагаю. С ATtiny85 все без проблем получилось, а вот с мегой328 никак не получается. LCD и светодиод подключены для отладки. Пробовал ручками в регистры значения прописывать - не помогло. Эти строки закоментарены. Может чего-то не хватает в коде?
Код: Выделить всё
/*******************************************************
Chip type               : ATmega328P
AVR Core Clock frequency: 8,000000 MHz
Memory model            : Small
External RAM size       : 0
Data Stack size         : 512
*******************************************************/

#include <mega328p.h>
#include <stdlib.h>
#include "lcdtwi.c"
#include <i2c.h>
#include <delay.h>
#include <sleep.h>

#define LED_GREEN PORTB.0       // светодиод

// Declare your global variables here
char flt_buf[16];   // буфер для вывода на LCD
unsigned int kw;

// Voltage Reference: AREF pin
#define ADC_VREF_TYPE ((0<<REFS1) | (1<<REFS0) | (0<<ADLAR) | (0<<MUX3) | (0<<MUX2) | (0<<MUX1) | (0<<MUX0))

// Watchdog timeout interrupt service routine
interrupt [WDT] void wdt_timeout_isr(void)
{
//        SMCR=(0<<SM2) | (1<<SM1) | (0<<SM0) | (0<<SE);

    sleep_disable();
} //interrupt [WDT]

// External Interrupt 0 service routine
interrupt [EXT_INT0] void ext_int0_isr(void)
{
// Нажата кнопка. Выходим из сна.
    sleep_disable();
} //interrupt [EXT_INT0]

// External Interrupt 1 service routine
interrupt [EXT_INT1] void ext_int1_isr(void)
{
//надо
} //interrupt [EXT_INT1]

// Timer1 overflow interrupt service routine
interrupt [TIM1_OVF] void timer1_ovf_isr(void)
{
//надо
} //interrupt [TIM1_OVF]

void main(void)
{
#pragma optsize-
// Crystal Oscillator division factor: 1
CLKPR=(1<<CLKPCE);
CLKPR=(0<<CLKPCE) | (0<<CLKPS3) | (0<<CLKPS2) | (0<<CLKPS1) | (0<<CLKPS0);
#ifdef _OPTIMIZE_SIZE_
#pragma optsize+
#endif


// Input/Output Ports initialization
// Port B initialization
// Function: Bit7=In Bit6=In Bit5=In Bit4=In Bit3=In Bit2=In Bit1=Out Bit0=Out
DDRB=(0<<DDB7) | (0<<DDB6) | (0<<DDB5) | (0<<DDB4) | (0<<DDB3) | (0<<DDB2) | (1<<DDB1) | (1<<DDB0);
// State: Bit7=T Bit6=T Bit5=T Bit4=T Bit3=T Bit2=T Bit1=0 Bit0=0
PORTB=(0<<PORTB7) | (0<<PORTB6) | (0<<PORTB5) | (0<<PORTB4) | (0<<PORTB3) | (0<<PORTB2) | (0<<PORTB1) | (0<<PORTB0);

// Port C initialization
// Function: Bit6=In Bit5=In Bit4=In Bit3=In Bit2=In Bit1=In Bit0=In
DDRC=(0<<DDC6) | (0<<DDC5) | (0<<DDC4) | (0<<DDC3) | (0<<DDC2) | (0<<DDC1) | (0<<DDC0);
// State: Bit6=T Bit5=T Bit4=T Bit3=T Bit2=T Bit1=T Bit0=T
PORTC=(0<<PORTC6) | (0<<PORTC5) | (0<<PORTC4) | (0<<PORTC3) | (0<<PORTC2) | (0<<PORTC1) | (0<<PORTC0);

// Port D initialization
// Function: Bit7=Out Bit6=Out Bit5=Out Bit4=Out Bit3=In Bit2=In Bit1=In Bit0=In
DDRD=(1<<DDD7) | (1<<DDD6) | (1<<DDD5) | (1<<DDD4) | (0<<DDD3) | (0<<DDD2) | (0<<DDD1) | (0<<DDD0);
// State: Bit7=0 Bit6=0 Bit5=0 Bit4=0 Bit3=T Bit2=T Bit1=T Bit0=T
PORTD=(0<<PORTD7) | (0<<PORTD6) | (0<<PORTD5) | (0<<PORTD4) | (0<<PORTD3) | (0<<PORTD2) | (0<<PORTD1) | (0<<PORTD0);

// Timer/Counter 0 initialization
// Clock source: System Clock
// Clock value: Timer 0 Stopped
// Mode: Normal top=0xFF
// OC0A output: Disconnected
// OC0B output: Disconnected
TCCR0A=(0<<COM0A1) | (0<<COM0A0) | (0<<COM0B1) | (0<<COM0B0) | (0<<WGM01) | (0<<WGM00);
TCCR0B=(0<<WGM02) | (0<<CS02) | (0<<CS01) | (0<<CS00);
TCNT0=0x00;
OCR0A=0x00;
OCR0B=0x00;

// Timer/Counter 1 initialization
// Clock source: System Clock
// Clock value: Timer1 Stopped
// Mode: Normal top=0xFFFF
// OC1A output: Disconnected
// OC1B output: Disconnected
// Noise Canceler: Off
// Input Capture on Falling Edge
// Timer1 Overflow Interrupt: Off
// Input Capture Interrupt: Off
// Compare A Match Interrupt: Off
// Compare B Match Interrupt: Off
TCCR1A=(0<<COM1A1) | (0<<COM1A0) | (0<<COM1B1) | (0<<COM1B0) | (0<<WGM11) | (0<<WGM10);
TCCR1B=(0<<ICNC1) | (0<<ICES1) | (0<<WGM13) | (0<<WGM12) | (0<<CS12) | (0<<CS11) | (0<<CS10);
TCNT1H=0x00;
TCNT1L=0x00;
ICR1H=0x00;
ICR1L=0x00;
OCR1AH=0x00;
OCR1AL=0x00;
OCR1BH=0x00;
OCR1BL=0x00;

// Timer/Counter 2 initialization
// Clock source: System Clock
// Clock value: Timer2 Stopped
// Mode: Normal top=0xFF
// OC2A output: Disconnected
// OC2B output: Disconnected
ASSR=(0<<EXCLK) | (0<<AS2);
TCCR2A=(0<<COM2A1) | (0<<COM2A0) | (0<<COM2B1) | (0<<COM2B0) | (0<<WGM21) | (0<<WGM20);
TCCR2B=(0<<WGM22) | (0<<CS22) | (0<<CS21) | (0<<CS20);
TCNT2=0x00;
OCR2A=0x00;
OCR2B=0x00;

// Timer/Counter 0 Interrupt(s) initialization
TIMSK0=(0<<OCIE0B) | (0<<OCIE0A) | (0<<TOIE0);

// Timer/Counter 1 Interrupt(s) initialization
TIMSK1=(0<<ICIE1) | (0<<OCIE1B) | (0<<OCIE1A) | (0<<TOIE1);

// Timer/Counter 2 Interrupt(s) initialization
TIMSK2=(0<<OCIE2B) | (0<<OCIE2A) | (0<<TOIE2);

// External Interrupt(s) initialization
// INT0: On
// INT0 Mode: Low level
// INT1: On
// INT1 Mode: Any change
// Interrupt on any change on pins PCINT0-7: Off
// Interrupt on any change on pins PCINT8-14: Off
// Interrupt on any change on pins PCINT16-23: Off
EICRA=(0<<ISC11) | (1<<ISC10) | (0<<ISC01) | (0<<ISC00);
EIMSK=(1<<INT1) | (1<<INT0);
EIFR=(1<<INTF1) | (1<<INTF0);
PCICR=(0<<PCIE2) | (0<<PCIE1) | (0<<PCIE0);

// USART initialization
// USART disabled
UCSR0B=(0<<RXCIE0) | (0<<TXCIE0) | (0<<UDRIE0) | (0<<RXEN0) | (0<<TXEN0) | (0<<UCSZ02) | (0<<RXB80) | (0<<TXB80);

// Analog Comparator initialization
// Analog Comparator: Off
// The Analog Comparator's positive input is
// connected to the AIN0 pin
// The Analog Comparator's negative input is
// connected to the AIN1 pin
ACSR=(1<<ACD) | (0<<ACBG) | (0<<ACO) | (0<<ACI) | (0<<ACIE) | (0<<ACIC) | (0<<ACIS1) | (0<<ACIS0);
// Digital input buffer on AIN0: On
// Digital input buffer on AIN1: On
DIDR1=(0<<AIN0D) | (0<<AIN1D);

// ADC initialization
// ADC Clock frequency: 125,000 kHz
// ADC Voltage Reference: AREF pin
// ADC Auto Trigger Source: Free Running
// Digital input buffers on ADC0: On, ADC1: Off, ADC2: Off, ADC3: Off
// ADC4: Off, ADC5: Off
DIDR0=(1<<ADC5D) | (1<<ADC4D) | (1<<ADC3D) | (1<<ADC2D) | (1<<ADC1D) | (0<<ADC0D);
ADMUX=ADC_VREF_TYPE;
ADCSRA=(1<<ADEN) | (0<<ADSC) | (0<<ADATE) | (0<<ADIF) | (0<<ADIE) | (1<<ADPS2) | (1<<ADPS1) | (1<<ADPS0);
ADCSRB=(0<<ADTS2) | (0<<ADTS1) | (0<<ADTS0);

// SPI initialization
// SPI disabled
SPCR=(0<<SPIE) | (0<<SPE) | (0<<DORD) | (0<<MSTR) | (0<<CPOL) | (0<<CPHA) | (0<<SPR1) | (0<<SPR0);

// TWI initialization
// TWI disabled
TWCR=(0<<TWEA) | (0<<TWSTA) | (0<<TWSTO) | (0<<TWEN) | (0<<TWIE);

// Watchdog Timer initialization
// Watchdog Timer Prescaler: OSC/1024k
// Watchdog timeout action: Interrupt
#pragma optsize-
WDTCSR=(0<<WDIF) | (0<<WDIE) | (1<<WDP3) | (1<<WDCE) | (0<<WDE) | (0<<WDP2) | (0<<WDP1) | (1<<WDP0);
WDTCSR=(1<<WDIF) | (1<<WDIE) | (1<<WDP3) | (0<<WDCE) | (0<<WDE) | (0<<WDP2) | (0<<WDP1) | (1<<WDP0);
#ifdef _OPTIMIZE_SIZE_
#pragma optsize+
#endif

// Global enable interrupts
#asm("sei")
I2C_Init();         //инициализируем TWI
LCD_ini();          //»нициализируем дисплей
   
while (1)
{

    // всхрапнем...
    for (kw=0; kw<5000; kw++)
    {
        LED_GREEN=1;
        delay_ms(50);
        LED_GREEN=0;

        clearlcd();
        setpos(0,0);
        sprintf(flt_buf, "kw=%u ", kw);
        str_lcd(flt_buf);
//        SMCR=(0<<SM2) | (1<<SM1) | (0<<SM0) | (1<<SE);
//        #asm("sleep")
        sleep_enable();
        powerdown();
    }

}//while
}//main
Вложения
fuse.png
fuse.png (22.23 КБ) Просмотров: 7022
Аватара пользователя
thor_nsk
Комсорг
 
Сообщения: 154
Зарегистрирован: 03 мар 2011, 08:12
Благодарил (а): 0 раз.
Поблагодарили: 1 раз.
Баллы репутации: 0

Re: ATMega328p и PowerDown

Сообщение thor_nsk » 05 ноя 2017, 07:54

И еще один вопрос. МК должен контролировать питание другой схемы. У всех персональное питание от литиевой батареи, но земля общая. Мне не понятно, почему в точке между R12 и R13 напряжение зависит от напряжения питания МК?! При изменении питания МК от 5 до 3.2 вольта (контролируемая батарея стабильна) напряжение в точке между R12 и R13 меняется в диапазоне 0.8 вольта. Пробовал ставить диоды (отмечено на схеме) - не помогло. Может не туда ставлю? Как тогда контролировать питание другой схемы? Делитель (R12, R13) нужен потому что возможен случай, что питание МК будет меньше чем у контролируемой схемы, а тогда АЦП будет выдавать 1023. Делитель через полевик подключается на время измерения напряжения, после этого отключается. Что-то у меня засада на ровном месте :(
Вложения
shema.jpg
shema.jpg (43.68 КБ) Просмотров: 7009
Аватара пользователя
thor_nsk
Комсорг
 
Сообщения: 154
Зарегистрирован: 03 мар 2011, 08:12
Благодарил (а): 0 раз.
Поблагодарили: 1 раз.
Баллы репутации: 0

Re: ATMega328p и PowerDown

Сообщение thor_nsk » 05 ноя 2017, 18:29

Пока ждал помощи пробовал решить проблему самостоятельно. С энергосбережением проблему решил. Засада оказалась вот в этом блоке:
Код: Выделить всё
WDTCSR=(0<<WDIF) | (0<<WDIE) | (1<<WDP3) | (1<<WDCE) | (0<<WDE) | (0<<WDP2) | (0<<WDP1) | (1<<WDP0);
WDTCSR=(1<<WDIF) | (1<<WDIE) | (1<<WDP3) | (0<<WDCE) | (0<<WDE) | (0<<WDP2) | (0<<WDP1) | (1<<WDP0);

А точнее в одном, блин, бите - WDE. Его надо было установить одновременно с WDCE и тогда делитель ватчдога установится иначе изменение игнорируется. Вот так правильно:
Код: Выделить всё
WDTCSR=(0<<WDIF) | (0<<WDIE) | (1<<WDP3) | (1<<WDCE) | (1<<WDE) | (0<<WDP2) | (0<<WDP1) | (1<<WDP0);
WDTCSR=(1<<WDIF) | (1<<WDIE) | (1<<WDP3) | (0<<WDCE) | (0<<WDE) | (0<<WDP2) | (0<<WDP1) | (1<<WDP0);

Второй вопрос, про напряжение на делителе, еще актуален.
Аватара пользователя
thor_nsk
Комсорг
 
Сообщения: 154
Зарегистрирован: 03 мар 2011, 08:12
Благодарил (а): 0 раз.
Поблагодарили: 1 раз.
Баллы репутации: 0

Re: ATMega328p и PowerDown

Сообщение thor_nsk » 09 ноя 2017, 05:25

Выяснил еще один не очень радостный момент. При отключении питания микроконтроллера, но при подключенной контролируемой батарее, микроконтроллер продолжал работать. Т.е. он питался от контролируемой батареи через АЦП :shock: Я такого совсем не ожидал. В общем, убрал делитель, убрал полевик, убрал одну батарею. Подцепил все в одну схему ибо задолбался. В такой конфигурации все работает. тему можно закрывать. Всем спасибо.
Аватара пользователя
thor_nsk
Комсорг
 
Сообщения: 154
Зарегистрирован: 03 мар 2011, 08:12
Благодарил (а): 0 раз.
Поблагодарили: 1 раз.
Баллы репутации: 0

Re: ATMega328p и PowerDown

Сообщение FELiS » 15 ноя 2017, 10:34

Внутри контроллера стоят защитные диоды катодом к VCC, через них и питается. АЦП скачет вследствие колебаний опорного напряжения на МК. Зачем мосфет в цепи акб?
Аватара пользователя
FELiS
Добрый Фелис
 
Сообщения: 532
Зарегистрирован: 11 янв 2007, 23:21
Благодарил (а): 27 раз.
Поблагодарили: 123 раз.
Баллы репутации: 156
ТехнарьТехнарь

Re: ATMega328p и PowerDown

Сообщение thor_nsk » 15 ноя 2017, 11:57

Через него МК должен был подключаться для контроля напряжения. И затем отключаться чтобы делитель R12-R13 не ел лишней энергии. У меня предполагалось иметь две схемы: 1 - управляющая и контролирующая питание, 2 - исполнительная. У каждой схемы свое питание. Не удалось осуществить задуманное и объединил обе схемы в одну питающуюся от одной батареи.
Аватара пользователя
thor_nsk
Комсорг
 
Сообщения: 154
Зарегистрирован: 03 мар 2011, 08:12
Благодарил (а): 0 раз.
Поблагодарили: 1 раз.
Баллы репутации: 0

Re: ATMega328p и PowerDown

Сообщение FELiS » 16 ноя 2017, 11:25

P-канальный фет надо. В разрыв плюса, затвор инвертировать. В твоей схеме потенциал истока плавает.
Аватара пользователя
FELiS
Добрый Фелис
 
Сообщения: 532
Зарегистрирован: 11 янв 2007, 23:21
Благодарил (а): 27 раз.
Поблагодарили: 123 раз.
Баллы репутации: 156
ТехнарьТехнарь


Вернуться в Техподдержка

Кто сейчас на конференции

Сейчас этот форум просматривают: нет зарегистрированных пользователей и гости: 73

cron
x

#{title}

#{text}