]> code.delx.au - pulseaudio/blob - src/pulsecore/proplist-util.c
bec857a4bbea752ab9d2e46b0fbae67a26456e2c
[pulseaudio] / src / pulsecore / proplist-util.c
1 /***
2 This file is part of PulseAudio.
3
4 Copyright 2008 Lennart Poettering
5
6 PulseAudio is free software; you can redistribute it and/or modify
7 it under the terms of the GNU Lesser General Public License as
8 published by the Free Software Foundation; either version 2.1 of the
9 License, or (at your option) any later version.
10
11 PulseAudio is distributed in the hope that it will be useful, but
12 WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 Lesser General Public License for more details.
15
16 You should have received a copy of the GNU Lesser General Public
17 License along with PulseAudio; if not, write to the Free Software
18 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
19 USA.
20 ***/
21
22 #ifdef HAVE_CONFIG_H
23 #include <config.h>
24 #endif
25
26 #include <string.h>
27 #include <locale.h>
28 #include <libintl.h>
29
30 #ifdef __APPLE__
31 #include <crt_externs.h>
32 #define environ (*_NSGetEnviron())
33 #elif !HAVE_DECL_ENVIRON
34 extern char **environ;
35 #endif
36
37 #include <pulse/gccmacro.h>
38 #include <pulse/proplist.h>
39 #include <pulse/utf8.h>
40 #include <pulse/xmalloc.h>
41 #include <pulse/util.h>
42
43 #include <pulsecore/core-util.h>
44
45 #if defined(HAVE_GLIB) && defined(PA_GCC_WEAKREF)
46 #include <glib.h>
47 static G_CONST_RETURN gchar* _g_get_application_name(void) PA_GCC_WEAKREF(g_get_application_name);
48 #endif
49
50 #if defined(HAVE_GTK) && defined(PA_GCC_WEAKREF)
51 #pragma GCC diagnostic ignored "-Wstrict-prototypes"
52 #include <gtk/gtk.h>
53 #include <gdk/gdkx.h>
54 static G_CONST_RETURN gchar* _gtk_window_get_default_icon_name(void) PA_GCC_WEAKREF(gtk_window_get_default_icon_name);
55 static Display *_gdk_display PA_GCC_WEAKREF(gdk_display);
56 #endif
57
58 #include "proplist-util.h"
59
60 static void add_glib_properties(pa_proplist *p) {
61
62 #if defined(HAVE_GLIB) && defined(PA_GCC_WEAKREF)
63
64 if (!pa_proplist_contains(p, PA_PROP_APPLICATION_NAME))
65 if (_g_get_application_name) {
66 const gchar *t;
67
68 /* We ignore the tiny race condition here. */
69
70 if ((t = _g_get_application_name()))
71 pa_proplist_sets(p, PA_PROP_APPLICATION_NAME, t);
72 }
73
74 #endif
75 }
76
77 static void add_gtk_properties(pa_proplist *p) {
78
79 #if defined(HAVE_GTK) && defined(PA_GCC_WEAKREF)
80
81 if (!pa_proplist_contains(p, PA_PROP_APPLICATION_ICON_NAME))
82 if (_gtk_window_get_default_icon_name) {
83 const gchar *t;
84
85 /* We ignore the tiny race condition here. */
86
87 if ((t = _gtk_window_get_default_icon_name()))
88 pa_proplist_sets(p, PA_PROP_APPLICATION_ICON_NAME, t);
89 }
90
91 if (!pa_proplist_contains(p, PA_PROP_WINDOW_X11_DISPLAY))
92 if (&_gdk_display && _gdk_display) {
93 const char *t;
94
95 /* We ignore the tiny race condition here. */
96
97 if ((t = DisplayString(_gdk_display)))
98 pa_proplist_sets(p, PA_PROP_WINDOW_X11_DISPLAY, t);
99 }
100
101 #endif
102 }
103
104 void pa_init_proplist(pa_proplist *p) {
105 char **e;
106 const char *pp;
107
108 pa_assert(p);
109
110 if (environ) {
111
112 /* Some applications seem to reset environ to NULL for various
113 * reasons, hence we need to check for this explicitly. See
114 * rhbz #473080 */
115
116 for (e = environ; *e; e++) {
117
118 if (pa_startswith(*e, "PULSE_PROP_")) {
119 size_t kl, skip;
120 char *k;
121 pa_bool_t override;
122
123 if (pa_startswith(*e, "PULSE_PROP_OVERRIDE_")) {
124 skip = 20;
125 override = TRUE;
126 } else {
127 skip = 11;
128 override = FALSE;
129 }
130
131 kl = strcspn(*e+skip, "=");
132
133 if ((*e)[skip+kl] != '=')
134 continue;
135
136 k = pa_xstrndup(*e+skip, kl);
137
138 if (!pa_streq(k, "OVERRIDE"))
139 if (override || !pa_proplist_contains(p, k))
140 pa_proplist_sets(p, k, *e+skip+kl+1);
141 pa_xfree(k);
142 }
143 }
144 }
145
146 if ((pp = getenv("PULSE_PROP"))) {
147 pa_proplist *t;
148
149 if ((t = pa_proplist_from_string(pp))) {
150 pa_proplist_update(p, PA_UPDATE_MERGE, t);
151 pa_proplist_free(t);
152 }
153 }
154
155 if ((pp = getenv("PULSE_PROP_OVERRIDE"))) {
156 pa_proplist *t;
157
158 if ((t = pa_proplist_from_string(pp))) {
159 pa_proplist_update(p, PA_UPDATE_REPLACE, t);
160 pa_proplist_free(t);
161 }
162 }
163
164 if (!pa_proplist_contains(p, PA_PROP_APPLICATION_PROCESS_ID)) {
165 char t[32];
166 pa_snprintf(t, sizeof(t), "%lu", (unsigned long) getpid());
167 pa_proplist_sets(p, PA_PROP_APPLICATION_PROCESS_ID, t);
168 }
169
170 if (!pa_proplist_contains(p, PA_PROP_APPLICATION_PROCESS_USER)) {
171 char *u;
172
173 if ((u = pa_get_user_name_malloc())) {
174 pa_proplist_sets(p, PA_PROP_APPLICATION_PROCESS_USER, u);
175 pa_xfree(u);
176 }
177 }
178
179 if (!pa_proplist_contains(p, PA_PROP_APPLICATION_PROCESS_HOST)) {
180 char *h;
181
182 if ((h = pa_get_host_name_malloc())) {
183 pa_proplist_sets(p, PA_PROP_APPLICATION_PROCESS_HOST, h);
184 pa_xfree(h);
185 }
186 }
187
188 if (!pa_proplist_contains(p, PA_PROP_APPLICATION_PROCESS_BINARY)) {
189 char *t;
190
191 if ((t = pa_get_binary_name_malloc())) {
192 char *c = pa_utf8_filter(t);
193 pa_proplist_sets(p, PA_PROP_APPLICATION_PROCESS_BINARY, c);
194 pa_xfree(t);
195 pa_xfree(c);
196 }
197 }
198
199 add_glib_properties(p);
200 add_gtk_properties(p);
201
202 if (!pa_proplist_contains(p, PA_PROP_APPLICATION_NAME)) {
203 const char *t;
204
205 if ((t = pa_proplist_gets(p, PA_PROP_APPLICATION_PROCESS_BINARY)))
206 pa_proplist_sets(p, PA_PROP_APPLICATION_NAME, t);
207 }
208
209 if (!pa_proplist_contains(p, PA_PROP_APPLICATION_LANGUAGE)) {
210 const char *l;
211
212 if ((l = setlocale(LC_MESSAGES, NULL)))
213 pa_proplist_sets(p, PA_PROP_APPLICATION_LANGUAGE, l);
214 }
215
216 if (!pa_proplist_contains(p, PA_PROP_WINDOW_X11_DISPLAY)) {
217 const char *t;
218
219 if ((t = getenv("DISPLAY"))) {
220 char *c = pa_utf8_filter(t);
221 pa_proplist_sets(p, PA_PROP_WINDOW_X11_DISPLAY, c);
222 pa_xfree(c);
223 }
224 }
225
226 if (!pa_proplist_contains(p, PA_PROP_APPLICATION_PROCESS_MACHINE_ID)) {
227 char *m;
228
229 if ((m = pa_machine_id())) {
230 pa_proplist_sets(p, PA_PROP_APPLICATION_PROCESS_MACHINE_ID, m);
231 pa_xfree(m);
232 }
233 }
234
235 if (!pa_proplist_contains(p, PA_PROP_APPLICATION_PROCESS_SESSION_ID)) {
236 char *s;
237
238 if ((s = pa_session_id())) {
239 pa_proplist_sets(p, PA_PROP_APPLICATION_PROCESS_SESSION_ID, s);
240 pa_xfree(s);
241 }
242 }
243 }
244
245 char *pa_proplist_get_stream_group(pa_proplist *p, const char *prefix, const char *cache) {
246 const char *r;
247 char *t;
248
249 if (!p)
250 return NULL;
251
252 if (cache && (r = pa_proplist_gets(p, cache)))
253 return pa_xstrdup(r);
254
255 if (!prefix)
256 prefix = "stream";
257
258 if ((r = pa_proplist_gets(p, PA_PROP_MEDIA_ROLE)))
259 t = pa_sprintf_malloc("%s-by-media-role:%s", prefix, r);
260 else if ((r = pa_proplist_gets(p, PA_PROP_APPLICATION_ID)))
261 t = pa_sprintf_malloc("%s-by-application-id:%s", prefix, r);
262 else if ((r = pa_proplist_gets(p, PA_PROP_APPLICATION_NAME)))
263 t = pa_sprintf_malloc("%s-by-application-name:%s", prefix, r);
264 else if ((r = pa_proplist_gets(p, PA_PROP_MEDIA_NAME)))
265 t = pa_sprintf_malloc("%s-by-media-name:%s", prefix, r);
266 else
267 t = pa_sprintf_malloc("%s-fallback:%s", prefix, r);
268
269 if (cache)
270 pa_proplist_sets(p, cache, t);
271
272 return t;
273 }