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

Цифровой Фильтр на ATmega

РАЗОБРАЛСЯ, тут еще на работе мне паренек здорово подсказал, так что большое спасибо всем за помощь.

Поделиться сообщением


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

Подскажите, пожалуйста. что представляет собой кольцевой буфер для входных значений фильтра (полосового)?

Поделиться сообщением


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

Подскажите, пожалуйста. что представляет собой кольцевой буфер для входных значений фильтра (полосового)?

Допустим Вы хотите сделать кольцевой буфер размера N на основе массива(можно еще на основе связанного списка, это просто - указатель последнего элемента должен указывать на первый, но он намного "тяжеловеснее" чем на основе массива). Функция которая отвечает за запись и/или чтение из буфера дожна при этом обеспечивать кольцевую адресацию. Тоесть достигнув элемента массива N-1(т.к. в C адресация начинается с "0") начать запись вновь с "0"-го элемента.

 

"Счетчик" сбрасывают многими конструкциями. if(i=N){i=0}, i=i%N или i=(i)&(N). Последний случай - это наложение маски. Работает очень быстро - но только для порядка из ряда целой степени двойки. Первый - предполагает ветвление - потому, как-бы, чисто теоретически - это не хорошо для кэша(а автомат-способ его работы у каждого процессора свой). Но лучше посмотреть, что наваял компилятор в ассемблерном листинге - т.е.есть ли ветвление вообще.

 

Для многих типов DSP можно инициализировать счетчик как кольцевой, он представляет из себя отдельный автомат - т.е. работает без затрат производительности.

 

Если любите смотреть на вещи не штампами и шаблонами, то можете попробовать сделать кольцевой автомат на основе таймера - тоже никаких затрат в виде тактов на проверки - но там нужно будет просчитать хватит ли Вам производительности такого фильтра - и это чисто asm решение, никакой переносимости...

 

P.S.: Сам кольцевой буфер проистекает от формулы свертки (это той, что по-идее должен знать каждый второкурсник ВТУЗа :biggrin::biggrin: :biggrin: ) - потому форма АЧХ(полосовой или какой иной) при этом никакой рояли не играет...

 

P.P.S.: исправил очепятку! Ибо знаю по-себе в какой ступор они приводят начинающих :biggrin: ...

Поделиться сообщением


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

"Счетчик" сбрасывают многими конструкциями. if(i=N){N=0}, i=i%N или i=(i)&(N). Последний случай - это наложение маски. Работает очень быстро - но только для порядка из ряда целой степени двойки.

Всё правильно. Только дополню, что существует особый случай, когда кольцевой буфер работает особенно изящно и быстро - это случай, когда он расчитан на 256 элементов. В этом случае, достаточно сделать "счетчик" цикла типа unsigned char, чтобы всё заработало само собой. Ведь в один байт число, большее чем 255 не запихнешь, поэтому такой счетчик будет "автоматически" сбрасываться при переполнении.

Кольцевые буферы на 256 элементов оказываются очень удобным инструментом не только для означенной цели, но и для буферизации приема и передачи по UART и USB, т.е. когда прием или получение приходят по прерыванию (UART) или когда отправлять по одному байту неэффективно (USB).

Примечание: у некоторых дурных компиляторов (кажется для MSP430) тип char трактуется как 2 байта, тогда ищите там другой однобайтный тип.

Поделиться сообщением


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

Примечание: у некоторых дурных компиляторов (кажется для MSP430) тип char трактуется как 2 байта, тогда ищите там другой однобайтный тип.
Это про CodeComposer Studio для серии TMS320.

Поделиться сообщением


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

Вообще кольцевой буфер легко реализуется, если его длина не только 256, но и 2n. Просто в адресе (смещении от начала массива) маскируете n младших бит

 

Про БИХ еще добавлю, если топикстартер еще не получил внятный ответ. Более-менее серьезные БИХ фильтры не возбуждаются только при расчетах в формате double. Это легко увидеть в Матлабе, введя округление до желаемого типа данных. А про целый формат и говорить нечего, только простейший аналог однозвенного RC-фильтра. Так что в целых - только КИХ-фильтры!

Изменено пользователем V_G

Поделиться сообщением


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

А однобайтный тип там есть?
Да какая разница? "Затачивание" на типы переменных, без явной проверки на выход за границы массива это потенциальный источник большого количества глюков программ. Именно такие уязвимости часто используют хакеры. Ваш совет считаю "вредным". В крайнем случае можно делать так, как V_G написал - маскировать старшие биты счетчика/индекса, урезая его разрядность. Но при этом остается проблема атомарности доступа к его значению.

Поделиться сообщением


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

При расчете коэффициентов полосового фильтра с помощью программы ciirf1, рекомендованной SasaVitebsk, и при расчете в MatLab получаются совершенно разные их значения , ciirf1 постоянно дает значение a1=0 и a3=0, почему?

Поделиться сообщением


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

Существуют разные алгоритмы синтеза фильтров и разные способы реализации одних и тех же требований к фильтрам. В том же Матлабе вы можете получить одинаковую амплитуду лепестков в полосе задержания (equiripple) или спадающую. Тут главное, чтобы результирующая ЧХ удовлетворяла. И чтобы было меньше негативное влияние округления (в Матлабе, повторю, можно увидеть отличия от расчетных характеристик при различных способах округления на различных этапах расчетов). Каким будет при этом a1, a3, не знаю. Есть одно строгое требование к коэффициентам КИХ: их сумма должна быть равна коэффициенту передачи на постоянном токе. Т.е. нулю для ФВЧ и ПФ.

Поделиться сообщением


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

Добавлю, что АЧХ фильтра и ФЧХ связаны между собой. В filterdesign MATLAB можно варьировать (оптимизировать) характеристики фильтра.

Поделиться сообщением


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

... рекомендованной SasaVitebsk ...

Боже упаси. :)

 

Я просто привёл, так сказать, для упрощения понимания. Существует большое количество других программ. Например QED. Они позволяют грамотно оценить искажения, импульсную характеристику фильтра и прочее. Кроме того, даже эта примитивная прога, даёт более точные значения коэффициентов в файл.

 

При выборе и обкатке своих фильтров я моделирую их поведение более сложно. Ну а на чёрную иногда ...

 

:)

 

Я просто привёл пример, чтобы начинающие программисты не боялись применять такую технологию как цифровая фильтрация.

Поделиться сообщением


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

...Существует большое количество других программ. Например QED...

 

Спасибо, очень интересный ресурс с QED. Кстати его адрес

http://www.kanecomputing.co.uk/mds_qedesign_lite.htm

 

Может не по теме, но там мощно..

http://www.kanecomputing.co.uk/

Поделиться сообщением


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

Всё правильно. Только дополню, что существует особый случай, когда кольцевой буфер работает особенно изящно и быстро - это случай, когда он расчитан на 256 элементов. В этом случае, достаточно сделать "счетчик" цикла типа unsigned char, чтобы всё заработало само собой.

 

Правильные пацаны работают прямо с указателями, а не индексами, и в таком случае даже буфер размещают с адреса, кратного 256, и тогда можно прямо младший байт указателя насиловать, не трогая старший. Ну или, если применяется операция and, то тоже можно выбрать правильный адрес начала буфера для уменьшения оверхеда. Второй совет, кстати, куда более универсальный, практически архитектурно-независимый.

Поделиться сообщением


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

Temp = (X0+X4+Y3-Y4)>>1; // все члены со сдвигом 4 = 1,7,9

Temp += (-X2-Y1); // ... 3 = 2,4 Естественно лучше записать Temp -= X2+Y1;

Temp >>= 1;

Temp += (-Y4); // .... 2 = 8 Опять таки Temp -= Y4;

Temp >>= 2; // Поскольку у нас нет членов со сдвигом >>1

Temp += (Y1<<1)-(Y2<<1)+Y3;

Если уж речь идёт о ATmega, то почему бы не использовать вместо этого извращения умножение командами mul...fmuls? Тогда при изменении коэффициентов не придётся каждый раз переписывать код программы. А небольшое увеличение времени (если оно ешё будет) тут, как я понял, совершенно не критично.

 

Я делал что-то подобное (не помню как оно научно называется) без обратных связей. Т.е. все элементы Y=0. Но с FIFO 256 слов. И умножал на коэффициенты в форме с плавающей запятой. Только была одна хитрость. Порядок у моих самодельных флоатов был представлен в виде кодов 0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80. Т.е., если сначала перемножить мантисы, а затем умножить командой mul полученную мантису на порядок, получалось собственно число.

Мне пришлось сделать таким образом т.к. коэффициенты были заранее неизвестны. Они вычислялись и уточнялись в процессе работы. И с быстродействием проблеммы были. Деление 16/16 из-за этого тоже пришлось переделать - меньше 60 тактов получилось.

Поделиться сообщением


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

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

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

Гость
Ответить в этой теме...

×   Вставлено с форматированием.   Вставить как обычный текст

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

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

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

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

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