เนื้อหาของบทความนำมาจากของฉัน
เครื่องตรวจจับโทนเสียง
ในที่สุด
ในสมัยก่อน เมื่อไม่ใช่ทุกครอบครัวจะมีทีวี และครึ่งหนึ่งของพวกเขาเปลี่ยนช่องโดยใช้คีม ข่าวที่น่าสนใจปรากฏในบทวิจารณ์ของสื่อมวลชนทางเทคนิคต่างประเทศว่าผู้ผลิตทีวีรายหนึ่งติดตั้งอุปกรณ์ของตนด้วยรีโมทคอนโทรลไร้สาย จากรายละเอียดเป็นที่ทราบกันว่ารีโมทคอนโทรลทำงานโดยไม่ใช้แบตเตอรี่เนื่องจากการใช้วิธีการที่ผิดปกติ - รีโมทคอนโทรลเป็นแบบกลไกและเป็นลูกผสมของเครื่องดนตรี - เมทัลโลโฟนและปืนพก ดรัมปืนลูกโม่บรรจุกระบอกโลหะที่มีความยาวต่างกัน และเมื่อหมุดยิงกระทบหนึ่งในนั้น กระบอกสูบก็เริ่มดังขึ้นด้วยความถี่ของมันเอง สงสัยจะอัลตราซาวนด์.. อุปกรณ์อิเล็กทรอนิกส์ในทีวีได้ยินสัญญาณนี้และเมื่อพิจารณาความถี่แล้วจึงดำเนินการที่เหมาะสม - สลับช่องเปลี่ยนระดับเสียงปิดทีวี
วันนี้เราจะพยายามสร้างระบบส่งคำสั่งนี้ขึ้นมาใหม่ โดยใช้ความรู้ของเราเกี่ยวกับมีเดียสตรีมเมอร์
เพื่อจำลองรีโมทคอนโทรล เราจะใช้ข้อความในตัวอย่างเครื่องกำเนิดเสียงของเรา เราจะเพิ่มการควบคุมความถี่ของเครื่องกำเนิดจากการกดแป้นพิมพ์และตัวรับพร้อมตัวถอดรหัสที่จะส่งออกคำสั่งที่ได้รับไปยังคอนโซล หลังจากการเปลี่ยนแปลง เครื่องกำเนิดควรสร้างโทนเสียง 6 ความถี่ ซึ่งเราจะเข้ารหัสคำสั่งเพื่อเพิ่ม/ลดระดับเสียง เปลี่ยนช่อง เปิด/ปิดทีวี ในการกำหนดค่าตัวตรวจจับ จะใช้โครงสร้างต่อไปนี้:
struct _MSToneDetectorDef{
char tone_name[8];
int frequency; /**<Expected frequency of the tone*/
int min_duration; /**<Min duration of the tone in milliseconds */
float min_amplitude; /**<Minimum amplitude of the tone, 1.0 corresponding to the normalized 0dbm level */
};
typedef struct _MSToneDetectorDef MSToneDetectorDef;
อุปกรณ์ตรวจจับสามารถกำหนดโครงสร้างเหล่านี้ได้ 10 โครงสร้าง ดังนั้นอุปกรณ์ตรวจจับหนึ่งตัวสามารถกำหนดค่าให้ตรวจจับสัญญาณทูโทนได้สิบสัญญาณ แต่เราจะใช้สัญญาณโทนเดียวเพียงหกสัญญาณเท่านั้น ในการถ่ายโอนการตั้งค่าไปยังเครื่องตรวจจับ จะใช้วิธีการ MS_TONE_DETECTOR_ADD_SCAN
เพื่อให้เครื่องตรวจจับแจ้งให้เราทราบว่าสัญญาณที่มีส่วนประกอบความถี่ที่ต้องการมาถึงอินพุตแล้ว เราต้องจัดเตรียมฟังก์ชันการโทรกลับที่จะเปิดใช้งานในกรณีนี้ ทำได้โดยใช้ฟังก์ชัน ms_filter_set_notify_callback(). อาร์กิวเมนต์จะได้รับตัวชี้ไปยังตัวกรอง ตัวชี้ไปยังฟังก์ชันการโทรกลับ และตัวชี้ไปยังข้อมูลที่เราต้องการส่งผ่านไปยังฟังก์ชันการโทรกลับ (ข้อมูลผู้ใช้)
เมื่อตัวตรวจจับถูกทริกเกอร์ ฟังก์ชันการโทรกลับจะได้รับข้อมูลผู้ใช้ ตัวชี้ไปยังตัวกรองตัวตรวจจับ ตัวระบุเหตุการณ์ และโครงสร้างที่อธิบายเหตุการณ์:
/** * Structure carried as argument of the MS_TONE_DETECTOR_EVENT**/
struct _MSToneDetectorEvent{
char tone_name[8]; /* Имя тона которое мы ему назначили при настройке детектора. */
uint64_t tone_start_time; /* Время в миллисекундах, когда тон был обнаружен. */
};
typedef struct _MSToneDetectorEvent MSToneDetectorEvent;
บล็อกไดอะแกรมของการประมวลผลสัญญาณแสดงอยู่ในรูปภาพชื่อ
ตอนนี้โค้ดโปรแกรมพร้อมความคิดเห็นแล้ว
/* Файл mstest4.c Имитатор пульта управления и приемника. */
#include <mediastreamer2/msfilter.h>
#include <mediastreamer2/msticker.h>
#include <mediastreamer2/dtmfgen.h>
#include <mediastreamer2/mssndcard.h>
#include <mediastreamer2/msvolume.h>
#include <mediastreamer2/mstonedetector.h>
/* Подключаем заголовочный файл с функциями управления событиями
* медиастримера. */
#include <mediastreamer2/mseventqueue.h>
/* Функция обратного вызова, она будет вызвана фильтром, как только он
* обнаружит совпадение характеристик входного сигнала с заданными. */
static void tone_detected_cb(void *data, MSFilter *f, unsigned int event_id,
MSToneDetectorEvent *ev)
{
printf(" Принята команда: %sn", ev->tone_name);
}
int main()
{
ms_init();
/* Создаем экземпляры фильтров. */
MSFilter *voidsource = ms_filter_new(MS_VOID_SOURCE_ID);
MSFilter *dtmfgen = ms_filter_new(MS_DTMF_GEN_ID);
MSFilter *volume = ms_filter_new(MS_VOLUME_ID);
MSSndCard *card_playback =
ms_snd_card_manager_get_default_card(ms_snd_card_manager_get());
MSFilter *snd_card_write = ms_snd_card_create_writer(card_playback);
MSFilter *detector = ms_filter_new(MS_TONE_DETECTOR_ID);
/* Очищаем массив находящийся внутри детектора тонов, он описывает
* особые приметы разыскиваемых сигналов.*/
ms_filter_call_method(detector, MS_TONE_DETECTOR_CLEAR_SCANS, 0);
/* Создаем источник тактов - тикер. */
MSTicker *ticker=ms_ticker_new();
/* Соединяем фильтры в цепочку. */
ms_filter_link(voidsource, 0, dtmfgen, 0);
ms_filter_link(dtmfgen, 0, volume, 0);
ms_filter_link(volume, 0, detector, 0);
ms_filter_link(detector, 0, snd_card_write, 0);
/* Подключаем к фильтру функцию обратного вызова. */
ms_filter_set_notify_callback(detector,
(MSFilterNotifyFunc)tone_detected_cb, NULL);
/* Подключаем источник тактов. */
ms_ticker_attach(ticker,voidsource);
/* Создаем массив, каждый элемент которого описывает характеристику
* одного из тонов, который требуется обнаруживать: Текстовое имя
* данного элемента, частота в герцах, длительность в миллисекундах,
* минимальный уровень относительно 0,775В. */
MSToneDetectorDef scan[6]=
{
{"V+", 440, 100, 0.1}, /* Команда "Увеличить громкость". */
{"V-", 540, 100, 0.1}, /* Команда "Уменьшить громкость". */
{"C+", 640, 100, 0.1}, /* Команда "Увеличить номер канала". */
{"C-", 740, 100, 0.1}, /* Команда "Уменьшить номер канала". */
{"ON", 840, 100, 0.1}, /* Команда "Включить телевизор". */
{"OFF", 940, 100, 0.1} /* Команда "Выключить телевизор". */
};
/* Передаем в детектор тонов приметы сигналов. */
int i;
for (i = 0; i < 6; i++)
{
ms_filter_call_method(detector, MS_TONE_DETECTOR_ADD_SCAN,
&scan[i]);
}
/* Настраиваем структуру, управляющую выходным сигналом генератора.*/
MSDtmfGenCustomTone dtmf_cfg;
dtmf_cfg.tone_name[0] = 0;
dtmf_cfg.duration = 1000;
dtmf_cfg.frequencies[0] = 440;
/* Будем генерировать один тон, частоту второго тона установим в 0.*/
dtmf_cfg.frequencies[1] = 0;
dtmf_cfg.amplitude = 1.0;
dtmf_cfg.interval = 0.;
dtmf_cfg.repeat_count = 0.;
/* Организуем цикл сканирования нажатых клавиш. Ввод нуля завершает
* цикл и работу программы. */
char key='9';
printf("Нажмите клавишу команды, затем ввод.n"
"Для завершения программы введите 0.n");
while(key != '0')
{
key = getchar();
if ((key >= 49) && (key <= 54))
{
printf("Отправлена команда: %cn", key);
/* Устанавливаем частоту генератора в соответствии с
* кодом нажатой клавиши.*/
dtmf_cfg.frequencies[0] = 440 + 100*(key-49);
/* Включаем звуковой генератор c обновленной частотой. */
ms_filter_call_method(dtmfgen, MS_DTMF_GEN_PLAY_CUSTOM,
(void*)&dtmf_cfg);
}
ms_usleep(20000);
}
}
เรารวบรวมและรันโปรแกรม หากทุกอย่างทำงานได้อย่างถูกต้อง หลังจากเปิดตัวแล้ว เราควรได้รับลักษณะการทำงานของโปรแกรมดังนี้:
$ ./mstest4
ALSA lib conf.c:4738:(snd_config_expand) Unknown parameters 0
ALSA lib control.c:954:(snd_ctl_open_noupdate) Invalid CTL default:0
ortp-warning-Could not attach mixer to card: Invalid argument
ALSA lib conf.c:4738:(snd_config_expand) Unknown parameters 0
ALSA lib pcm.c:2266:(snd_pcm_open_noupdate) Unknown PCM default:0
ALSA lib conf.c:4738:(snd_config_expand) Unknown parameters 0
ALSA lib pcm.c:2266:(snd_pcm_open_noupdate) Unknown PCM default:0
ortp-warning-Strange, sound card Intel 82801AA-ICH does not seems to be capable of anything, retrying with plughw...
Нажмите клавишу команды, затем ввод.
Для завершения программы введите 0.
ortp-warning-alsa_set_params: periodsize:256 Using 256
ortp-warning-alsa_set_params: period:8 Using 8
กดปุ่มใดก็ได้ตั้งแต่ "1" ถึง "6" เพื่อยืนยันด้วยปุ่ม "Enter" คุณจะเห็นรายการลักษณะนี้:
2
Отправлена команда: 2
Принята команда: V-
1
Отправлена команда: 1
Принята команда: V+
3
Отправлена команда: 3
Принята команда: C+
4
Отправлена команда: 4
Принята команда: C-
0
$
เราเห็นว่าส่งเสียงคำสั่งสำเร็จแล้วและตัวตรวจจับตรวจจับได้
ในบทความถัดไป เราจะหันมาส่งสัญญาณเสียงผ่านเครือข่ายอีเธอร์เน็ตโดยใช้โปรโตคอล RTP และนำไปใช้กับรีโมทคอนโทรลของเราทันที
ที่มา: will.com