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

"неравномерно" Идущие Часы - Проблема С Контроллером


Виталик1705

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

Добрый день! Собрал часы по инструкции из статьи http://habrahabr.ru/post/130594/. К проекту прилагается только исходный код, поэтому откомпилировал в Code::Blocks, прошил с помощью USBasp. Но вот беда, часы идут очень медленно (приблизительно раз в 10 медленнее реального времени). Сначала думал кварц не тот, проверил в Протеусе - то же самое. Думаю ошибка в программе, но мои знания в AVR весьма поверхностны, что бы ее найти. Смотрю на код - вроде все нормально, а на деле - не работает. Помогите, кто чем может. Заранее спасибо.

#include <avr/io.h>
#include <avr/sleep.h>
#include <avr/interrupt.h>

// таблица длительностей интервалов (в тиках)
const uint8_t intervals [] __attribute__ ((progmem)) =
{
0x01, 0x05, 0x04, 0x04,
0x05, 0x09, 0x04, 0x02,
0x09, 0x02, 0x02, 0x07,
0x02, 0x03, 0x03, 0x03,

0x03, 0x05, 0x05, 0x07,
0x04, 0x06, 0x07, 0x01,
0x02, 0x01, 0x05, 0x02,
0x03, 0x03, 0x0C, 0x04,

0x06, 0x02, 0x01, 0x0D,
0x02, 0x03, 0x04, 0x03,
0x01, 0x01, 0x06, 0x04,
0x06, 0x04, 0x02, 0x06,

0x02, 0x04, 0x06, 0x01,
0x07, 0x04, 0x02, 0x02,
0x01, 0x05, 0x03, 0x04
};

ISR (TIM0_COMPA_vect) // Прерывание по таймеру
{
static uint8_t polarity =0;
static uint8_t tick_no =0;
static uint8_t next_tick_no =0;
static uint8_t interval_no =0;

if (tick_no == next_tick_no)
{

if (polarity)
{ // положительный импульс
PORTB |= (1 << 2)|(1 << 4);
}
else
{ // отрицательный импульс
PORTB |= (1 << 0)|(1 << 1);
}
polarity ^= 0x01;

if (interval_no < 59)
{
next_tick_no += intervals [interval_no];
interval_no++;
}
else
{ // началась новая минута
next_tick_no = 0;
interval_no = 0;
}
}

if (tick_no < 239)
tick_no ++;
else
tick_no = 0;
}

ISR (TIM0_COMPB_vect) // Второе прерывание, через 100 мс после первого
{ // отключаем двигатель
PORTB &= ~((1 << 0)|(1 << 1)|(1 << 2)|(1 << 4));
}

void main(void) __attribute__((noreturn));

void main(void)
{
DDRB |= (1 << 0)|(1 << 1)|(1 << 2)|(1 << 4);
TCCR0A = (1 << WGM01); // режим работы таймера "сброс по совпадению"
TCCR0B = (1 << CS02); // предделитель таймера 1/256
OCR0A = 0x1F; // Верхняя граница счета
// 32768 / 256 / 0x20 = 4 Hz - частота следования прерываний
OCR0B = 0x0С; // примерно 100 мс от сброса счетчика до второго прерывания
TIMSK0 = (1 << OCIE0A)|(1 << OCIE0B); // разрешить прерывания по таймеру
set_sleep_mode (SLEEP_MODE_IDLE);
sei (); // глобальное разрешение прерываний
for (;
sleep_enable (); // в спячку!
}

Изменено пользователем Виталик1705
Ссылка на комментарий
Поделиться на другие сайты

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

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

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

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

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

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

Подскажите а в какой програме все эт написано? Под какие програмы этот код вообще подходит?

Не знаю, но я компилировал в Code::Blocks и в AVR Studio, на выходе hex-файлы одинаковые. Это от компилятора зависит. Возможно в этом причина?
Ссылка на комментарий
Поделиться на другие сайты

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

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

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

Еще я заметил что интервал между 1 и вторым прерыванием ровно 100 миллисекунд, как и полагается. Но первое прерывание возникает очень редко.

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

  • 3 недели спустя...

Расписал сегодня алгоритм обработчика прерывания на бумаге. Все нормально работает. Почему он может не выполнятся в контроллере и симуляторе?

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

Интересно, этот сайт http://dxportal.ru/drugie-novosti-radiolyubitelya/23231-34neravnomerno34-iduschie-chasy-problema-s-kontrollerom.html имеет какое-либо отношение к сайту cxem.net? Там мой вопрос с этого форума, но я его задавал только на этом форуме.

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

Разобрался в программе. Вот код:

#include <avr/pgmspace.h>
#include <avr/io.h>
#include <avr/sleep.h>
#include <avr/interrupt.h>

// таблица длительностей интервалов (в тиках)
//const uint8_t intervals []__attribute__ ((progmem)) =
const uint8_t intervals [] PROGMEM =
{
   0x01, 0x05, 0x04, 0x04,
   0x05, 0x09, 0x04, 0x02,
   0x09, 0x02, 0x02, 0x07,
   0x02, 0x03, 0x03, 0x03,
   0x03, 0x05, 0x05, 0x07,
   0x04, 0x06, 0x07, 0x01,
   0x02, 0x01, 0x05, 0x02,
   0x03, 0x03, 0x0C, 0x04,
   0x06, 0x02, 0x01, 0x0D,
   0x02, 0x03, 0x04, 0x03,
   0x01, 0x01, 0x06, 0x04,
   0x06, 0x04, 0x02, 0x06,
   0x02, 0x04, 0x06, 0x01,
   0x07, 0x04, 0x02, 0x02,
   0x01, 0x05, 0x03, 0x04
};

ISR (TIM0_COMPA_vect)   // Прерывание по таймеру
{
   static uint8_t polarity =0;
   static uint8_t tick_no =0;
   static uint8_t next_tick_no =0;
   static uint8_t interval_no =0;
   if (tick_no == next_tick_no)
   {
    if (polarity)
    {
	    // положительный импульс
	    PORTB |= (1 << 2)|(1 << 4);
    }
    else
    {
	    // отрицательный импульс
	    PORTB |= (1 << 0)|(1 << 1);
    }
    polarity ^= 0x01;
    if (interval_no < 59)
    {
	    //next_tick_no += intervals[interval_no];
	    next_tick_no += pgm_read_byte(&intervals[interval_no]);
	    interval_no++;
    }
    else
    {
	    // началась новая минута
	    next_tick_no = 0;
	    interval_no = 0;
    }
   }
   if (tick_no < 239)
   {
   tick_no ++;
   }
   else
   {
    tick_no = 0;
   }
}
ISR (TIM0_COMPB_vect)   // Второе прерывание, через 100 мс после первого
{
   // отключаем двигатель
   PORTB &= ~((1 << 0)|(1 << 1)|(1 << 2)|(1 << 4));
}
int main(void) __attribute__((noreturn));
int main(void)
{
   DDRB |= (1 << 0)|(1 << 1)|(1 << 2)|(1 << 4);
   TCCR0A = (1 << WGM01);  // режим работы таймера "сброс по совпадению"
   TCCR0B =(1 << CS02);   // предделитель таймера 1/256
   OCR0A = 0x1f;				  // Верхняя граница счета 1f
// 32768 / 256 / 0x20 = 4 Hz - частота следования прерываний
   OCR0B =  0x0C;  // примерно 100 мс от сброса счетчика до второго прерывания
   TIMSK0 = (1 << OCIE0A)|(1 << OCIE0B); // разрешить прерывания по таймеру
   set_sleep_mode (SLEEP_MODE_IDLE);
   sei ();		 // глобальное разрешение прерываний
   for (;
    sleep_enable ();	    // в спячку!
}

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

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

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

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

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

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

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

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

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

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

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

    • А что тогда стабилизируется? Если нужен "реостат" в смысле "регулируемое сопротивление" то  нужна электронная нагрузка с режимом "постоянное сопротивление" или CR. Элемент нагрузки в любом случае это ИТУН (даже в вашей схеме), надо просто ввести обратную связь с коллектора, т.к. I=U/R, I - ток, задаваемый нагрузочному элементу, U - напряжение на нагрузке, а R - желаемое сопротивление. Что-то типа такого.   Транзистору с отсутствием напряжения на коллекторе ничего не будет, база просто диодом, через который течет ток, определяемый положением движка и управляющим напряжением. Отсутствие напряжения на базе при наличии напряжения на коллекторе не критично до тех пор, пока база притянута к земле через сопротивление (в отечественных даташитах это параметр максимального напряжения коллектор-эмиттер при заданном сопротивлении база-эмиттер). Это сильно критично для старых транзисторов с большим током утечки коллектора, если его не пустить с базы в эмиттер через сопротивление меньшее или равное примерно 0,25/Ik0, транзистор начнёт приоткрываться. Для современных транзисторов сопротивление может быть относительно большим из-за малых утечек. Да, странный вопрос. Если нужно сопротивление, то зачем схема источника тока
    • На лицо полное непонимание принципа работы биполярного транзистора.
    • ..совсем точно, - выключаешь БП, выпаиваешь R1 и на его место впаиваешь резистор сопротивлением 3 кОм.. включаешь БП и вертишь R2 - проверяешь (измеряешь вольтметром или мультиметром) диапазон изменения напряжения на выходе БП..  если не устраивает опять меняешь R1..   более подробно будет сложнее..
    • Так, ну ладно,... забудем про сигнатуры.... И так,... как я вижу Программу. Программа состоит из двух файлов: экзешника и хелпера, а лучше одного экзешника, чтобы не путаться в обилии файлов в директории. Можно выбрать номер кома и его скорость, тип МК, а также другие вспомогательные парметры,..... и ... большая красная кнопка "сгенерировать Бутлодер", при нажатии которой, создается файл Бутлодера, с соответствующим именем. Внутри файла Бутлодера несколько байт выделено для названия МК, которые возможно считать Программой, и которые в дальнейшем определяют модель работы Программы с Бутлодером, чтобы не путать пользоваетля. Сделать относительно короткими пакеты обмена между Программой и Бутлодером, чтобы избежать длительных зависаний. Конечно это увеличит общее время записи, ну мы же никуда не спешим. Добавить команду считывания названия МК из Бутлодера, что кажется мне самым важным. Да, как я организую процесс записи флеш: заполняю буфер несколькими командами, запускаю команду записать страницу. МК принимает данную команду, декодирует и запускает стирание страницы, дожидается окончания процесса стирания страницы (<5 мс, ЦПУ остановлено), запускает запись страницы (<5 мс, ЦПУ остановлено), осуществляет проверку записанных данных на странице(верификация) и посылает ответ ПК с флагом проверки данных.(Это все из дата шита). Запись в еепром производится побайтно в каждой комманде. (ну мы же никуда не спешим). При записи в еепром считываются данные из ячейки, и если они совпадают с теми, что нужно записать, то запись не осуществляется. В противном случае запускаем запись данных в ячейку (<5 мс), ждем окончания записи , осуществляем проверку записанных данных в еепром(верификация) и посылаем ответ ПК с флагом проверки данных. Ну как-то так крупными мазками.     И да, существует два типа Бутлодеров, написанных с применением прирываний, и без прирываний. объем без прерываний как ни странно меньше.
    • Не эта конкретная плата, а AMP2X15 на основе TPA3110D2: Могли бы в своей теме задать вопрос, а не плодить темы.
    • Не хватает. Это же песочница? Да, только бы понять это точнее. Спасибо за идею.
  • Похожий контент

×
×
  • Создать...