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

Декодирование OOK сигнала на stm8s


carlogulliani

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

Добрый день, пытаюсь разобраться с декодированием сигнала ASK/OOK на stm8s003. В OOK главную роль играют период сигнала, так, например, 0 - 400 мкс, 1 - 800 мкс. +- 10%. Поэтому основная для меня проблема считать количество микросекунд между событиями, в ардуино за это отвечает функция micros()

Включаю внутреннее тактирование 16МГц, настраиваю таймер TIM1 на частоту 1MHz (1 микросекунда), у таймера стоит прерывание по переполнению, и когда счетчик доходит до предела, я в некую переменную добавляю максимальное значение счетчика, на выводе вижу не те значения, которые ожидаю, также попробовал считать через capture compare, тоже безрезультатно

 

#include "stm8s.h"
#include "timer.h"
#include "gpio.h"

volatile uint32_t tick = 0;

//Interruption
ISR_TIM1(TIM1_OVF) {
  tick += 65535;
    TIM1_SR1 &= ~0x01;     // clear all interrupt flags
    TIM1_SR1 = 0;
    TIM1_SR2 = 0;
    TIM1_CR1 = 0;
    TIM1_CCER1 &= ~0X11; // STOP TIMER
    //TIM1_SR1 = 0;
    //PinToggle(PORTD, PIN5);
}

uint8_t flag = 0;

ISR_TIM1_CC(TIM1_CAP_COM) {
    if (TIM1_SR1 & 0x01) {
        TIM1_SR1 = 0;
        TIM1_CR1 = 0;
        TIM1_CCER1 &= ~0x11;
        if (TIM1_SR2) {
            TIM1_SR2 = 0;
            flag = 1;
        }
        else {
            flag = 0;
        }
    }
    if (TIM1_SR1 & 0x02) {
        TIM1_SR1 &= ~0x02;
    }
}


uint32_t systick() {
    return (uint32_t)(TIM1_CNTRH << 8) | TIM1_CNTRL;
}

uint32_t millis() {
    return tick + (systick() / (F_CPU / 1000));
}

uint32_t micros() {
    return tick + (systick() / (F_CPU / 1000000ul));
}

void tim1_init(void) {
    // 1MHz = 1uS
    TIM1_PSCRH = 0x00;
    TIM1_PSCRL = 0x15;

    TIM1_IER = 0x04 | 0x02 | 0x01; // Enable interrups
    TIM1_CCMR1 = 1;
    TIM1_CCMR2 = 2;
    TIM1_CCER1 = 0x20;
    TIM1_SMCR = 0x54;
}

void tim1_start() {
    TIM1_CNTRH = 0;
    TIM1_CNTRL = 0;
    TIM1_CCR2H = 0;
    TIM1_CCR2L = 0;
    TIM1_CCER1 |= 0x11;
    TIM1_CR1 = 0x01;
}

uint16_t tim1_pulse_len() {
    return ((uint16_t)(TIM1_CCR2H << 8) | TIM1_CCR2L);
}

Принимаю сигнал

 

ISR_PORTD(handlerD) {
	state = PinRead(GD0_PORT, GD0_PIN);
  uint32_t m = micros();
  if (state == HIGH) {
    lolen = m - prevtime;
    printf("//");
  }
  else {
    hilen = m - prevtime;
    printf("\\");
  }
  prevtime = m;
  if (state == HIGH)
  {
    if (flag == 0) {
      // the end of packet
      if (CheckValue(Pe, hilen) && CheckValue(Pe2, lolen)) // valid 1
      {
        printf("OK-PE\r\n");
        if (bcounter < 32)
          code1 = (code1 << 1) | 1;
        else if (bcounter < 64)
          code2 = (code2 << 1) | 1;
        bcounter++;
      }
      else if (CheckValue(Pe2, hilen) && CheckValue(Pe, lolen)) // valid 0
      {
        printf("OK-PE2\r\n");
        if (bcounter < 32)
          code1 = (code1 << 1) | 0;
        else if (bcounter < 64)
          code2 = (code2 << 1) | 0; bcounter++;
      }
      else
        bcounter = 0;
    }
    else {
      if (bcounter < 32)
          code1 = (code1 << 1) | 1;
        else if (bcounter < 64)
          code2 = (code2 << 1) | 1;
        bcounter++;
    }
  }
  //printf("bcounter: %d\r\n", bcounter);
  if (bcounter >= 65)
  {
    printf("Got: %x %x\r\n", code1, code2);
    Pe2 = lolen;
    Pe = hilen;
    flag = 1;
    bcounter = 0;
    code1 = 0;
    code2 = 0;
  }
}

 

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

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

31 minutes ago, carlogulliani said:

настраиваю таймер TIM1 на частоту 1MHz

Если таймер настроен на 1МГц, то systick() и есть время в микросекундах,  то есть в прерывании от порта просто считывайте показания таймера:

ISR_PORTD(handlerD) 

{
	state = PinRead(GD0_PORT, GD0_PIN);
	uint16_t m = systick();
	if (state == HIGH) 
	{ 
		lolen = m - prevtime;
	}
	prevtime = m;

Таймер 16-битный, все переменные uint16_t.

А зачем таймер останавливать? Пусть себе крутится.

Зачем прерывание по переполнению?

Зачем прерывание по Capture/Compare?

Переменная tick не нужна, просто вычитайте предыдущее значение из  текущего, и всё будет пучком.

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

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

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

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

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

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

Just now, Yurkin2015 said:

Если таймер настроен на 1МГц, то systick() и есть время в микросекундах,  то есть в прерывании от порта просто считывайте показания таймера:


ISR_PORTD(handlerD) 

{
	state = PinRead(GD0_PORT, GD0_PIN);
	uint16_t m = systick();
	if (state == HIGH) 
	{ 
		lolen = m - prevtime;
	}
	prevtime = m;

Таймер 16-битный, все переменные uint16_t.

А зачем таймер останавливать? Пусть себе крутится.

Зачем прерывание по переполнению?

Зачем прерывание по Capture/Compare?

Спасибо за ваш комментарий! Прерывание по переполнению сделал для того, чтобы, когда досчитает то предельного значения (в данном случае до 65535) и обнулится, не получилось так, что при первом вызове значения, условно будут, 60000, а во втором 14000, то есть дельта тогда будет не правильно считаться. CС хотел использовать для подсчета периода сигнала, условно, когда первый раз дрыгнулась нога и флаг 0, включаю таймер, как только прошел период - считаю значения СС. Про остановку таймера, вы правы, незачем.

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

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

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

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

2 minutes ago, carlogulliani said:

когда досчитает то предельного значения

 

2 minutes ago, carlogulliani said:

60000, а во втором 14000

Переполнение не помешает. Если переменные типа uint16_t, то всё прекрасно вычтется 14000 - 60000 = 19536, и получится правильный результат.

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

36 minutes ago, Yurkin2015 said:

 

Переполнение не помешает. Если переменные типа uint16_t, то всё прекрасно вычтется 14000 - 60000 = 19536, и получится правильный результат.

Спасибо

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

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

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

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

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

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

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

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

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

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

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

    • О, это очень полезные регистры! в 88 только GPIOR0 сохранил свои полезные свойства. использую их как флаги событий прерываний. для GPIOR0 адрес порта ввода-вывода 0х1Е, а значит к нему применяются команды cbi, sbi, sbic, sbis   ну и   in, out. Когда происходит прерывание, процессор переходит на адрес обработки прерывания, вот там-то мы и располагаем код: sbi   GPIOR0, 0     ;установить в 1 бит 0 в регистре GPIOR0 reti                        ;вернуться из прерывания   Без использования регистра GPIOR0, а с использованием обычного регистра код выглядел бы иначе: push   R0                          ;освобождаем регистр R0 для SREG и сохраняем его in        R0, SREG               ;сохраняем SREG в R0, все флаги операций текущей программы sbr     R23, 1<<0             ;выставляем флаг признака прерывания, например бит 0 в регистре R23 out    SREG, R0               ;восстанавливаем SREG, все флаги операций текущей программы pop   R0                          ;восстанавливаем значение R0 reti                                  ;вернуться из прерывания   Нетрудно заметить......!  
    • Сабсоник 3 порядка потом усилитель на Оу. Далее все на столе отстроить и все 
    • Вот и я думаю сделать на сдвоенном операционнике входной усилитель и сабсоник. 
    • Я всегда подозревал, что эта схема была содрана кЕтайцами с какого-то старого и хорошо известного (но не у нас) блока питания, и что в оригинале использовались именно 741 операционники. И вот тому подтверждение... Все равно те микросхемы и транзисторы что они используют в наборах, чаще всего подделки. Мне например, пришлось заменить D1047 транзистор что шел в наборе на пару таких же, но нормальных (выдраных с дохлого усилка). Транзистор из набора (маркировка явно "левая", без какого-либо намека на изготовителя) сильно грелся даже на 1,5А. Такой же транзистор D1047, но из усилка грелся раза в два меньше, да и маркировка "нормальная".  Подозреваю что и TL081 что в наборе идут тоже возможно что перемаркированные 741 (они супердешевые, сравнимо с 358). 741 операционники выпускали все кому не лень, аналогов было выпущено очень много за полвека.  Были и на плюс-минус 22в, надо смотреть конкретный даташит и производителя, даже от буквы в конце это зависит.  Например, есть такой аналог uA741 от ST ("микро-А741"). ua741-957400.pdf ua741.pdf MA741.PDF
    • Нет. Эта модель TRI для 3фаз. Да и вся разводка сделана для 3х фаз. EVOLUTION Mono ➞ 1~ 230V ± 10% 50/60Гц EVOLUTION Tri ➞ 3~ 400V ± 10% 50/60Гц В конце второй минуты есть внутренности, подключение и включение пульта. Можно заметить, что платы идентичные. 
    • У меня такой усь был, он никакой. 
    • Приветствую , вопрос по светодиодам HL5-10 .  В схеме они по 1.65в каждый , автор на вегалабе упомянул суммарное напряжение в каждом плече 4.7в и что допустимо применение стабилитронов.   Вопрос какое напряжение допустимо и за что отвечает? В наличии светики только 1.7в.  Насколько оправданно установка стабилитронов ?  Схему прилагаю
×
×
  • Создать...