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

Декодирование 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 пользователей онлайн

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