Обсуждаем контроллеры компании Atmel.
Ответить

Re: Ампервольтметр на ATmega8

Сб дек 30, 2023 13:31:22

veso74 писал(а):ISR(ADC_vect) имеет более низкий приоритет чем ISR(TIMER0_OVF_vect). Т.е. если программа находится в первом фрагменте кода, то индикация иногда прерывает расчет.
глупость ты сказал.
работа АЦП и работа таймера не зависят друг от друга.
а обработать результат измерения можно без проблем после окончания прерывания от таймера.

Re: Ампервольтметр на ATmega8

Сб дек 30, 2023 13:33:56

ISR(ADC_vect) имеет более низкий приоритет чем ISR(TIMER0_OVF_vect). Т.е. если программа находится в первом фрагменте кода, то индикация иногда прерывает расчет.

При чем здесь приоритеты? У ТС ведь нет вложенных прерываний.
В прерываниях здесь вообще нет необходимости - только дополнительные хлопоты и возможные ошибки.
Несколько переменных не объявлены волатильными, хотя изменяются в обработчике...
В вообще, код - страх и ужас(. И так оно и работает...

Re: Ампервольтметр на ATmega8

Сб дек 30, 2023 13:46:39

Считайте как ошибка машинного перевода. Имел в виду избегать прерывание в прерывании, с чем ATmega8 не справляется, в отличие от других микроконтроллеров.
Последний раз редактировалось veso74 Сб дек 30, 2023 16:33:57, всего редактировалось 1 раз.

Re: Ампервольтметр на ATmega8

Сб дек 30, 2023 14:24:34

поставил конденсатор на 100 нФ на AREF -показания стали плавать больше. Реально хуже...

Добавлено after 5 minutes 41 second:
ISR(ADC_vect) имеет более низкий приоритет чем ISR(TIMER0_OVF_vect). Т.е. если программа находится в первом фрагменте кода, то индикация иногда прерывает расчет.

При чем здесь приоритеты? У ТС ведь нет вложенных прерываний.
В прерываниях здесь вообще нет необходимости - только дополнительные хлопоты и возможные ошибки.
Несколько переменных не объявлены волатильными, хотя изменяются в обработчике...
В вообще, код - страх и ужас(. И так оно и работает...

Я не претендую на звание синьора. Учусь, написал как смог, потому здесь и спрашиваю, так как понимаю, что сделано не совсем по феншую. Делал по примерам другого кода из инета. Скажу больше, большинство кода, которое просмотрел, когда разбирался с АЦП так и построено.

Re: Ампервольтметр на ATmega8

Сб дек 30, 2023 18:53:44

Последний проект был совсем не давно, надо было считать сопротивление с платинового датчика РТ100, и вывести его на лсд с помощью атмеги8, с точностью 1 градус ( Цельсия естественно). И всё получилось .

Добавлено after 8 minutes 17 seconds:
Изображение
Это датчик рт100 в кипящем чайнике

Добавлено after 23 minutes 20 seconds:
А это тот же датчик, только в парУ над кипящей водой


Изображение

Re: Ампервольтметр на ATmega8

Сб дек 30, 2023 20:08:39

Последний проект был совсем не давно, надо было считать сопротивление с платинового датчика РТ100, и вывести его на лсд с помощью атмеги8, с точностью 1 градус ( Цельсия естественно). И всё получилось .

Добавлено after 8 minutes 17 seconds:
Изображение
Это датчик рт100 в кипящем чайнике

Добавлено after 23 minutes 20 seconds:
А это тот же датчик, только в парУ над кипящей водой


Изображение

Кодом поделитесь?, посмотреть, может чего-то нового понять...

Re: Ампервольтметр на ATmega8

Сб дек 30, 2023 21:02:48

Код не тайна )))

Добавлено after 32 seconds:
ща поищу ,выложу !

Добавлено after 2 minutes 55 seconds:
Спойлер
Код:
/*******************************************************
Chip type               : ATmega8
Program type            : Application
AVR Core Clock frequency: 1,000000 MHz
*******************************************************/
#include <mega8.h>
#include <delay.h>
#include <alcd.h>
unsigned char i;
long int tem;
long int a;
//long int  i=0,i1=0;
//long  int y,z,x2;
//int mass[26]={10000,10390,10779,11167,11554,11940,12324,12707,13089,13470,13850,14229,14606,14982,15358,15731,16104,16476,16846,17216,17584,17951,18317,18682,19045,19407} ;
interrupt [TIM1_OVF] void timer1_ovf_isr(void)
{
// Place your code here

}

interrupt [TIM2_OVF] void timer2_ovf_isr(void)
{
// Place your code here

}

// Voltage Reference: AREF pin
#define ADC_VREF_TYPE ((0<<REFS1) | (0<<REFS0) | (0<<ADLAR))

// Read the AD conversion result
unsigned int read_adc(unsigned char adc_input)
{
ADMUX=adc_input | ADC_VREF_TYPE;
// Delay needed for the stabilization of the ADC input voltage
delay_us(10);
// Start the AD conversion
ADCSRA|=(1<<ADSC);
// Wait for the AD conversion to complete
while ((ADCSRA & (1<<ADIF))==0);
ADCSRA|=(1<<ADIF);
return ADCW;
}
      void lcd_pr (long int te,unsigned char i)
                         {
                                     if (te>=1000) goto la1;
                                      if (te>=100) goto la2;  // отброс незначащих нулей                                                    // тоже отброс незначащих нулей
                                       if (te>=10)goto la3;
                                       goto la4 ;
                             la1:           lcd_putchar(te/1000+48);      // сотни
                             la2:           lcd_putchar(te%1000/100+48);  // десятки
                             la3:           lcd_putchar((te%100)/10+48);  // единицы
                                if (i==1)  lcd_putchar(44);              // запятая
                             la4:           lcd_putchar(te%10+48);      // десятые градуса
                         }
void main(void)
{
// Declare your local variables here

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

// Port C initialization
// Function: Bit6=In Bit5=Out Bit4=In Bit3=In Bit2=In Bit1=In Bit0=In
DDRC=(0<<DDC6) | (1<<DDC5) | (0<<DDC4) | (0<<DDC3) | (0<<DDC2) | (0<<DDC1) | (0<<DDC0);
// State: Bit6=T Bit5=0 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=In Bit6=In Bit5=In Bit4=Out Bit3=Out Bit2=Out Bit1=Out Bit0=Out
DDRD=(0<<DDD7) | (0<<DDD6) | (0<<DDD5) | (1<<DDD4) | (1<<DDD3) | (1<<DDD2) | (1<<DDD1) | (1<<DDD0);
// State: Bit7=T Bit6=T Bit5=T Bit4=0 Bit3=0 Bit2=0 Bit1=0 Bit0=0
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
TCCR0=(0<<CS02) | (0<<CS01) | (0<<CS00);
TCNT0=0x00;

// Timer/Counter 1 initialization
// Clock source: System Clock
// Clock value: 15,625 kHz
// Mode: Normal top=0xFFFF
// OC1A output: Disconnected
// OC1B output: Disconnected
// Noise Canceler: Off
// Input Capture on Falling Edge
// Timer Period: 4,1943 s
// Timer1 Overflow Interrupt: On
// 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) | (1<<CS11) | (1<<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
// OC2 output: Disconnected
ASSR=0<<AS2;
TCCR2=(0<<PWM2) | (0<<COM21) | (0<<COM20) | (0<<CTC2) | (0<<CS22) | (0<<CS21) | (0<<CS20);
TCNT2=0x00;
OCR2=0x00;

// Timer(s)/Counter(s) Interrupt(s) initialization
TIMSK=(0<<OCIE2) | (1<<TOIE2) | (0<<TICIE1) | (0<<OCIE1A) | (0<<OCIE1B) | (1<<TOIE1) | (0<<TOIE0);

// External Interrupt(s) initialization
// INT0: Off
// INT1: Off
MCUCR=(0<<ISC11) | (0<<ISC10) | (0<<ISC01) | (0<<ISC00);

// USART initialization
// USART disabled
UCSRB=(0<<RXCIE) | (0<<TXCIE) | (0<<UDRIE) | (0<<RXEN) | (0<<TXEN) | (0<<UCSZ2) | (0<<RXB8) | (0<<TXB8);

// 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);

// ADC initialization
// ADC Clock frequency: 31,250 kHz
// ADC Voltage Reference: AREF pin
ADMUX=ADC_VREF_TYPE;
ADCSRA=(1<<ADEN) | (0<<ADSC) | (0<<ADFR) | (0<<ADIF) | (0<<ADIE) | (1<<ADPS2) | (0<<ADPS1) | (1<<ADPS0);
SFIOR=(0<<ACME);

// 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);

// Alphanumeric LCD initialization
// Connections are specified in the
// Project|Configure|C Compiler|Libraries|Alphanumeric LCD menu:
// RS - PORTD Bit 3
// RD - PORTB Bit 1
// EN - PORTD Bit 4
// D4 - PORTD Bit 2
// D5 - PORTD Bit 1
// D6 - PORTD Bit 0
// D7 - PORTC Bit 5
// Characters/line: 8
lcd_init(8);

// Global enable interrupts
#asm("sei")
     
   lcd_gotoxy(0,0); 
   lcd_putsf(" Pt-100");
    lcd_gotoxy(0,1);
    lcd_putsf("Alex_641");
    delay_ms(2000);
    lcd_init(8);
   lcd_gotoxy(0,0);   
   lcd_putsf("R=");
   lcd_gotoxy(0,1);   
   lcd_putsf("T=");   
   
 
     
while (1)
      {
      a=0;
      for (i=1;i<129;i++)
      {
     
       tem=read_adc(6)-1;   //считываем ацп
       delay_ms(10);
       a=a+tem;
       }
       tem=a>>7;//среднее арифметическое 128 показаний
        a=(195*tem)/100;  // переводим ацп в сопротивление
         lcd_gotoxy(2,0);
         lcd_pr (a,1);
           lcd_putsf(" ");
      tem=(a*1000-1000000)/385;  // переводим сопротивление в температуру (умноженную на 10)
   
       if (tem%10>5) tem=tem+10; // если последняя цифра больше 5 округляем вверх
        tem=tem/10;     // получаем температуру округлённую до целых
         lcd_gotoxy(2,1);
        if (a>=1000 && a<=1980)
        {
       
          lcd_pr (tem,0);
          }
          else
          {
          if (a<1000) { lcd_putsf("<0    ");lcd_gotoxy(4,1);}
          if (a>1980)  lcd_putsf(">254");
          }
           lcd_putchar(223);
           lcd_putsf("C  ");
             
           // delay_ms(100);
      }
}

Re: Ампервольтметр на ATmega8

Сб дек 30, 2023 22:35:55

veso74 писал(а): Имел в виду избегать прерывание в прерывании, с чем ATmega8 не справляется
зачем избегать? и АТмега8 справится и без проблем. просто в обработчике прерывания нужно сразу разрешить прерывания, и тогда другое прерывание прервёт этот обработчик.

Re: Ампервольтметр на ATmega8

Сб дек 30, 2023 23:22:41

Starichok51, разве в АВРах возможны вложенные прерывания? По-моему, вы что-то перепутали.

Re: Ампервольтметр на ATmega8

Вс дек 31, 2023 07:55:01

>TEHb<, специально для тебя повторю:
если в обработчике прерывания разрешить прерывания, тогда другое прерывание прервёт этот обработчик.
Ответить