Ср сен 29, 2021 11:40:58
/*******************************************************
This program was created by the
CodeWizardAVR V3.14 Advanced
Automatic Program Generator
© Copyright 1998-2014 Pavel Haiduc, HP InfoTech s.r.l.
http://www.hpinfotech.com
Project : test_5
Version :
Date : 24.09.2021
Author :
Company :
Comments:
Chip type : ATmega8A
Program type : Application
AVR Core Clock frequency: 1,000000 MHz
Memory model : Small
External RAM size : 0
Data Stack size : 256
*******************************************************/
#include <mega8.h>
#include <inttypes.h>
//#include <avr/io.h>
#include <interrupt.h>
#include <delay.h>
volatile uint8_t period = 3;
volatile uint8_t pulsewidth = 125;
volatile uint16_t icr1_temp;
// ADC interrupt service routine
ISR(TIM1_OVF){
icr1_temp = (uint16_t) ((uint32_t) ((uint32_t)period * 0x10000)/256);
/*if(icr1_temp < 850){ //Без этого куска кода МК виснет при значении ICR1 < 850
icr1_temp = 850;
}*/
ICR1 = icr1_temp;
OCR1A = (uint16_t) ((uint32_t) ((uint32_t)pulsewidth * ICR1)/256);
}
ISR(ADC_INT){
if(ADMUX & (1<<MUX0)){
pulsewidth = ADCH; //get ch 1
ADMUX = (0<<REFS1)|(1<<REFS0)|(1<<ADLAR) | (1<<MUX1) | (0<<MUX0); //set ch 0
}else{
period = ADCH;
PORTD = period; //get ch 0
ADMUX = (0<<REFS1)|(1<<REFS0)|(1<<ADLAR) | (1<<MUX1) | (1<<MUX0); //set ch 1
}
//ICR1 = (uint16_t) ((uint32_t) ((uint32_t)period * 0x10000)/256);
//OCR1A = (uint16_t) ((uint32_t) ((uint32_t)pulsewidth * ICR1)/256);
//delay_us(10);
// Start the AD conversion
ADCSRA|=(1<<ADSC);
}
void main(void)
{
// Write your code here
// Input/Output Ports initialization
// Port B initialization
// Function: Bit1=Out
DDRB = (1<<DDB1);
// State: Bit1=0
PORTB = (0<<PORTB1);
// Port D initialization
// Function: Bit7=In Bit6=In Bit5=In Bit4=In Bit3=In Bit2=In Bit1=In Bit0=In
DDRD=(1<<DDD7) | (1<<DDD6) | (1<<DDD5) | (1<<DDD4) | (1<<DDD3) | (1<<DDD2) | (1<<DDD1) | (1<<DDD0);
// Timer/Counter 1 initialization
// Clock source: System Clock
// Clock value: ...... kHz
// Mode: Ph. & fr. cor. PWM top=ICR1
// OC1A output: Non-Inverted PWM
// OC1B output: Disconnected
// Noise Canceler: Off
// Input Capture on Falling Edge
// Timer Period: 0 us
// Output Pulse(s):
// OC1A Period: 0 us
// Timer1 Overflow Interrupt: Off
// Input Capture Interrupt: Off
// Compare A Match Interrupt: Off
// Compare B Match Interrupt: Off
TCCR1A=(1<<COM1A1) | (0<<COM1A0) | (0<<COM1B1) | (0<<COM1B0) | (0<<WGM11) | (0<<WGM10);
TCCR1B=(0<<ICNC1) | (0<<ICES1) | (1<<WGM13) | (0<<WGM12) | (0<<CS12) | (0<<CS11) | (1<<CS10);
TCNT1=0x00;
ICR1=0x0300;
OCR1A=0x00;
OCR1B=0x00;
// Timer(s)/Counter(s) Interrupt(s) initialization
TIMSK=(0<<OCIE2) | (0<<TOIE2) | (0<<TICIE1) | (0<<OCIE1A) | (0<<OCIE1B) | (1<<TOIE1) | (0<<TOIE0);
// Analog Comparator: Off
ACSR=(1<<ACD) | (0<<ACBG) | (0<<ACO) | (0<<ACI) | (0<<ACIE) | (0<<ACIC) | (0<<ACIS1) | (0<<ACIS0);
// ADC initialization
// ADC Clock frequency: ..... kHz
// ADC Voltage Reference: AVCC pin
// Only the 8 most significant bits of
// the AD conversion result are used
ADMUX = (0<<REFS1) | (1<<REFS0) | (1<<ADLAR);
ADCSRA=(1<<ADEN) | (0<<ADSC) | (0<<ADFR) | (0<<ADIF) | (1<<ADIE) | (1<<ADPS2) | (1<<ADPS1) | (1<<ADPS0);
SFIOR=(0<<ACME);
ADCSRA |= (1<<ADSC); // Start the AD conversion
sei();
while (1){
}
}
Ср сен 29, 2021 13:53:05
Ср сен 29, 2021 16:12:45
Ср сен 29, 2021 18:38:56
Ср сен 29, 2021 19:03:59
Starichok51 писал(а):а чтобы быстро сделать работу в прерывании по переполнению, нужно вообще исключить любые вычисления в прерывании.
Ср сен 29, 2021 19:22:42
тебе это таймер нужен не для 16 бит, а для того, чтобы можно было записать любое число в качестве периода ШИМ.Kalisnik писал(а):Я делаю ШИМ на 16-ти разрядном таймере Т1. Максимальная величина ICR1 = 0xffff (65 535).
Ср сен 29, 2021 20:18:08
Ср сен 29, 2021 20:46:02
volatile uint16_t period;
volatile uint16_t pulsewidth;
ISR(TIMER1_OVF_vect){
ICR1 = period;
OCR1A = pulsewidth;
}
#include <util/atomic.h>
uint16_t period_temp, pulsewidth_temp;
uint16_t period_temp2, pulsewidth_temp2;
#define ADC_VREF_TYPE ((0<<REFS1) | (1<<REFS0) | (1<<ADLAR))
while (1)
{
ADMUX = ADC_VREF_TYPE | (0<<MUX0);//set ch 0
_delay_us(10);
ADCSRA|=(1<<ADSC);
while ((ADCSRA & (1<<ADIF))==0);
ADCSRA|=(1<<ADIF);
pulsewidth_temp = ADCH;
ADMUX = ADC_VREF_TYPE | (1<<MUX0);//set ch 1
_delay_us(10);
ADCSRA|=(1<<ADSC);
while ((ADCSRA & (1<<ADIF))==0);
ADCSRA|=(1<<ADIF);
period_temp = ADCH;
period_temp2 = (uint16_t) ((uint32_t) ((uint32_t)period_temp * 0x10000)/256);
pulsewidth_temp2 = (uint16_t) ((uint32_t) ((uint32_t)pulsewidth_temp * (uint32_t)period_temp2)/256);
ATOMIC_BLOCK(ATOMIC_RESTORESTATE)
{ // блок кода с запрещенными прерываниями
period = period_temp2;
pulsewidth = pulsewidth_temp2;
}
}
Ср сен 29, 2021 21:05:37
Ср сен 29, 2021 21:19:42
Да, вы правы, при 0x10000 он в ICR1L не будет писать, там будет 0.Starichok51 писал(а):ICR1H = period;
ICR1L = 0;
Ср сен 29, 2021 21:35:45
о чем ты думал, когда это писал?Dimon456 писал(а):При тактовой частоте 1МГц, ICR1H=1 ICR1L=0, 1953Гц
При тактовой частоте 1МГц, ICR1H=2 ICR1L=0, 651Гц
Ср сен 29, 2021 21:38:18
Ср сен 29, 2021 22:06:58
Малость ошибся.Starichok51 писал(а):о чем ты думал, когда это писал?
При тактовой частоте 1МГц, ICR1H=1 ICR1L=0, 1953Гц
При тактовой частоте 1МГц, ICR1H=2 ICR1L=0, 976Гц
При тактовой частоте 8МГц, ICR1H=1 ICR1L=0, 15625Гц
При тактовой частоте 8МГц, ICR1H=2 ICR1L=0, 7812Гц
Ср сен 29, 2021 22:08:30
Ср сен 29, 2021 22:21:30
Starichok51 писал(а):и не 255, а 256 у тебя будет шаг изменения регистра ICR1.
Ср сен 29, 2021 22:56:28
Ср сен 29, 2021 23:09:52
Dimon456 писал(а):и тогда получится
ICR1H = period;
ICR1L = 0;
Ср сен 29, 2021 23:31:57
Ср сен 29, 2021 23:34:51
Чт сен 30, 2021 08:33:14