-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathsounddetect.c
137 lines (104 loc) · 4.02 KB
/
sounddetect.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
/* GStreamer
* Copyright (C) 2000,2001,2002,2003,2005
* Thomas Vander Stichele <thomas at apestaart dot org>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
/* compile:
gcc sounddetect.c -o sounddetect -lm `pkg-config --cflags --libs gstreamer-1.0`
*/
#include <string.h>
#include <math.h>
#define GLIB_DISABLE_DEPRECATION_WARNINGS
#include <gst/gst.h>
static gboolean
message_handler (GstBus * bus, GstMessage * message, gpointer data)
{
if (message->type == GST_MESSAGE_ELEMENT) {
const GstStructure *s = gst_message_get_structure (message);
const gchar *name = gst_structure_get_name (s);
if (strcmp (name, "level") == 0) {
gint channels;
GstClockTime endtime;
gdouble rms_dB, peak_dB, decay_dB;
gdouble rms;
const GValue *array_val;
const GValue *value;
GValueArray *rms_arr, *peak_arr, *decay_arr;
gint i;
// if (!gst_structure_get_clock_time (s, "endtime", &endtime))
// g_warning ("Could not parse endtime");
/* the values are packed into GValueArrays with the value per channel */
// array_val = gst_structure_get_value (s, "rms");
// rms_arr = (GValueArray *) g_value_get_boxed (array_val);
array_val = gst_structure_get_value (s, "peak");
peak_arr = (GValueArray *) g_value_get_boxed (array_val);
// array_val = gst_structure_get_value (s, "decay");
// decay_arr = (GValueArray *) g_value_get_boxed (array_val);
/* we can get the number of channels as the length of any of the value
* arrays */
// channels = rms_arr->n_values;
// g_print ("endtime: %" GST_TIME_FORMAT ", channels: %d\n",
// GST_TIME_ARGS (endtime), channels);
// g_print ("channel %d\n", i);
// value = g_value_array_get_nth (rms_arr, i);
// rms_dB = g_value_get_double (value);
value = g_value_array_get_nth (peak_arr, 0);
peak_dB = g_value_get_double (value);
// g_print("peak: %f dB\n", peak_dB);
g_print("%f\n", peak_dB);
// value = g_value_array_get_nth (decay_arr, i);
// decay_dB = g_value_get_double (value);
// g_print (" RMS: %f dB, peak: %f dB, decay: %f dB\n",
// rms_dB, peak_dB, decay_dB);
/* converting from dB to normal gives us a value between 0.0 and 1.0 */
// rms = pow (10, rms_dB / 20);
// g_print (" normalized rms value: %f\n", rms);
}
}
return TRUE;
}
int
main (int argc, char *argv[])
{
GstElement *audiosrc, *audioconvert, *level, *fakesink;
GstElement *pipeline;
GstCaps *caps;
GstBus *bus;
guint watch_id;
GMainLoop *loop;
gst_init (&argc, &argv);
GError *error = NULL;
gchar *audio_device = argv[1];
gchar *pl = g_strconcat (
"alsasrc device=hw:", audio_device, " ! audioconvert ! level post-messages=true interval=500000000 ! fakesink sync=true",
NULL);
pipeline = gst_parse_launch (pl, &error);
g_free(pl);
if (error) {
g_printerr ("Failed to parse launch: %s\n", error->message);
return -1;
}
bus = gst_element_get_bus (pipeline);
watch_id = gst_bus_add_watch (bus, message_handler, NULL);
gst_element_set_state (pipeline, GST_STATE_PLAYING);
/* we need to run a GLib main loop to get the messages */
loop = g_main_loop_new (NULL, FALSE);
g_main_loop_run (loop);
g_source_remove (watch_id);
g_main_loop_unref (loop);
return 0;
}