default: esd is obsolete, hence don't load it anymore by default
[pulseaudio-mirror.git] / src / pulsecore / core-scache.c
blob4f2a44dc2d55dc453f895bd1022ebb8d5d52e8a8
1 /***
2 This file is part of PulseAudio.
4 Copyright 2004-2008 Lennart Poettering
5 Copyright 2006 Pierre Ossman <ossman@cendio.se> for Cendio AB
7 PulseAudio is free software; you can redistribute it and/or modify
8 it under the terms of the GNU Lesser General Public License as published
9 by the Free Software Foundation; either version 2.1 of the License,
10 or (at your option) any later version.
12 PulseAudio is distributed in the hope that it will be useful, but
13 WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 General Public License for more details.
17 You should have received a copy of the GNU Lesser General Public License
18 along with PulseAudio; if not, write to the Free Software
19 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
20 USA.
21 ***/
23 #ifdef HAVE_CONFIG_H
24 #include <config.h>
25 #endif
27 #include <stdlib.h>
28 #include <stdio.h>
29 #include <sys/types.h>
30 #include <dirent.h>
31 #include <sys/stat.h>
32 #include <errno.h>
33 #include <limits.h>
34 #include <time.h>
36 #ifdef HAVE_GLOB_H
37 #include <glob.h>
38 #endif
40 #ifdef HAVE_WINDOWS_H
41 #include <windows.h>
42 #endif
44 #include <pulse/mainloop.h>
45 #include <pulse/channelmap.h>
46 #include <pulse/timeval.h>
47 #include <pulse/util.h>
48 #include <pulse/volume.h>
49 #include <pulse/xmalloc.h>
50 #include <pulse/rtclock.h>
52 #include <pulsecore/sink-input.h>
53 #include <pulsecore/play-memchunk.h>
54 #include <pulsecore/core-subscribe.h>
55 #include <pulsecore/namereg.h>
56 #include <pulsecore/sound-file.h>
57 #include <pulsecore/core-rtclock.h>
58 #include <pulsecore/core-util.h>
59 #include <pulsecore/log.h>
60 #include <pulsecore/core-error.h>
61 #include <pulsecore/macro.h>
63 #include "core-scache.h"
65 #define UNLOAD_POLL_TIME (60 * PA_USEC_PER_SEC)
67 static void timeout_callback(pa_mainloop_api *m, pa_time_event *e, const struct timeval *t, void *userdata) {
68 pa_core *c = userdata;
70 pa_assert(c);
71 pa_assert(c->mainloop == m);
72 pa_assert(c->scache_auto_unload_event == e);
74 pa_scache_unload_unused(c);
76 pa_core_rttime_restart(c, e, pa_rtclock_now() + UNLOAD_POLL_TIME);
79 static void free_entry(pa_scache_entry *e) {
80 pa_assert(e);
82 pa_namereg_unregister(e->core, e->name);
83 pa_subscription_post(e->core, PA_SUBSCRIPTION_EVENT_SAMPLE_CACHE|PA_SUBSCRIPTION_EVENT_REMOVE, e->index);
84 pa_xfree(e->name);
85 pa_xfree(e->filename);
86 if (e->memchunk.memblock)
87 pa_memblock_unref(e->memchunk.memblock);
88 if (e->proplist)
89 pa_proplist_free(e->proplist);
90 pa_xfree(e);
93 static pa_scache_entry* scache_add_item(pa_core *c, const char *name) {
94 pa_scache_entry *e;
96 pa_assert(c);
97 pa_assert(name);
99 if ((e = pa_namereg_get(c, name, PA_NAMEREG_SAMPLE))) {
100 if (e->memchunk.memblock)
101 pa_memblock_unref(e->memchunk.memblock);
103 pa_xfree(e->filename);
104 pa_proplist_clear(e->proplist);
106 pa_assert(e->core == c);
108 pa_subscription_post(c, PA_SUBSCRIPTION_EVENT_SAMPLE_CACHE|PA_SUBSCRIPTION_EVENT_CHANGE, e->index);
109 } else {
110 e = pa_xnew(pa_scache_entry, 1);
112 if (!pa_namereg_register(c, name, PA_NAMEREG_SAMPLE, e, TRUE)) {
113 pa_xfree(e);
114 return NULL;
117 e->name = pa_xstrdup(name);
118 e->core = c;
119 e->proplist = pa_proplist_new();
121 pa_idxset_put(c->scache, e, &e->index);
123 pa_subscription_post(c, PA_SUBSCRIPTION_EVENT_SAMPLE_CACHE|PA_SUBSCRIPTION_EVENT_NEW, e->index);
126 e->last_used_time = 0;
127 pa_memchunk_reset(&e->memchunk);
128 e->filename = NULL;
129 e->lazy = FALSE;
130 e->last_used_time = 0;
132 pa_sample_spec_init(&e->sample_spec);
133 pa_channel_map_init(&e->channel_map);
134 pa_cvolume_init(&e->volume);
135 e->volume_is_set = FALSE;
137 pa_proplist_sets(e->proplist, PA_PROP_MEDIA_ROLE, "event");
139 return e;
142 int pa_scache_add_item(
143 pa_core *c,
144 const char *name,
145 const pa_sample_spec *ss,
146 const pa_channel_map *map,
147 const pa_memchunk *chunk,
148 pa_proplist *p,
149 uint32_t *idx) {
151 pa_scache_entry *e;
152 char st[PA_SAMPLE_SPEC_SNPRINT_MAX];
153 pa_channel_map tmap;
155 pa_assert(c);
156 pa_assert(name);
157 pa_assert(!ss || pa_sample_spec_valid(ss));
158 pa_assert(!map || (pa_channel_map_valid(map) && ss && pa_channel_map_compatible(map, ss)));
160 if (ss && !map) {
161 pa_channel_map_init_extend(&tmap, ss->channels, PA_CHANNEL_MAP_DEFAULT);
162 map = &tmap;
165 if (chunk && chunk->length > PA_SCACHE_ENTRY_SIZE_MAX)
166 return -1;
168 if (!(e = scache_add_item(c, name)))
169 return -1;
171 pa_sample_spec_init(&e->sample_spec);
172 pa_channel_map_init(&e->channel_map);
173 pa_cvolume_init(&e->volume);
174 e->volume_is_set = FALSE;
176 if (ss) {
177 e->sample_spec = *ss;
178 pa_cvolume_reset(&e->volume, ss->channels);
181 if (map)
182 e->channel_map = *map;
184 if (chunk) {
185 e->memchunk = *chunk;
186 pa_memblock_ref(e->memchunk.memblock);
189 if (p)
190 pa_proplist_update(e->proplist, PA_UPDATE_REPLACE, p);
192 if (idx)
193 *idx = e->index;
195 pa_log_debug("Created sample \"%s\" (#%d), %lu bytes with sample spec %s",
196 name, e->index, (unsigned long) e->memchunk.length,
197 pa_sample_spec_snprint(st, sizeof(st), &e->sample_spec));
199 return 0;
202 int pa_scache_add_file(pa_core *c, const char *name, const char *filename, uint32_t *idx) {
203 pa_sample_spec ss;
204 pa_channel_map map;
205 pa_memchunk chunk;
206 int r;
207 pa_proplist *p;
209 #ifdef OS_IS_WIN32
210 char buf[MAX_PATH];
212 if (ExpandEnvironmentStrings(filename, buf, MAX_PATH))
213 filename = buf;
214 #endif
216 pa_assert(c);
217 pa_assert(name);
218 pa_assert(filename);
220 p = pa_proplist_new();
221 pa_proplist_sets(p, PA_PROP_MEDIA_FILENAME, filename);
223 if (pa_sound_file_load(c->mempool, filename, &ss, &map, &chunk, p) < 0) {
224 pa_proplist_free(p);
225 return -1;
228 r = pa_scache_add_item(c, name, &ss, &map, &chunk, p, idx);
229 pa_memblock_unref(chunk.memblock);
230 pa_proplist_free(p);
232 return r;
235 int pa_scache_add_file_lazy(pa_core *c, const char *name, const char *filename, uint32_t *idx) {
236 pa_scache_entry *e;
238 #ifdef OS_IS_WIN32
239 char buf[MAX_PATH];
241 if (ExpandEnvironmentStrings(filename, buf, MAX_PATH))
242 filename = buf;
243 #endif
245 pa_assert(c);
246 pa_assert(name);
247 pa_assert(filename);
249 if (!(e = scache_add_item(c, name)))
250 return -1;
252 e->lazy = TRUE;
253 e->filename = pa_xstrdup(filename);
255 pa_proplist_sets(e->proplist, PA_PROP_MEDIA_FILENAME, filename);
257 if (!c->scache_auto_unload_event)
258 c->scache_auto_unload_event = pa_core_rttime_new(c, pa_rtclock_now() + UNLOAD_POLL_TIME, timeout_callback, c);
260 if (idx)
261 *idx = e->index;
263 return 0;
266 int pa_scache_remove_item(pa_core *c, const char *name) {
267 pa_scache_entry *e;
269 pa_assert(c);
270 pa_assert(name);
272 if (!(e = pa_namereg_get(c, name, PA_NAMEREG_SAMPLE)))
273 return -1;
275 pa_assert_se(pa_idxset_remove_by_data(c->scache, e, NULL) == e);
277 pa_log_debug("Removed sample \"%s\"", name);
279 free_entry(e);
281 return 0;
284 void pa_scache_free_all(pa_core *c) {
285 pa_scache_entry *e;
287 pa_assert(c);
289 while ((e = pa_idxset_steal_first(c->scache, NULL)))
290 free_entry(e);
292 if (c->scache_auto_unload_event) {
293 c->mainloop->time_free(c->scache_auto_unload_event);
294 c->scache_auto_unload_event = NULL;
298 int pa_scache_play_item(pa_core *c, const char *name, pa_sink *sink, pa_volume_t volume, pa_proplist *p, uint32_t *sink_input_idx) {
299 pa_scache_entry *e;
300 pa_cvolume r;
301 pa_proplist *merged;
302 pa_bool_t pass_volume;
304 pa_assert(c);
305 pa_assert(name);
306 pa_assert(sink);
308 if (!(e = pa_namereg_get(c, name, PA_NAMEREG_SAMPLE)))
309 return -1;
311 merged = pa_proplist_new();
312 pa_proplist_sets(merged, PA_PROP_MEDIA_NAME, name);
313 pa_proplist_sets(merged, PA_PROP_EVENT_ID, name);
315 if (e->lazy && !e->memchunk.memblock) {
316 pa_channel_map old_channel_map = e->channel_map;
318 if (pa_sound_file_load(c->mempool, e->filename, &e->sample_spec, &e->channel_map, &e->memchunk, merged) < 0)
319 goto fail;
321 pa_subscription_post(c, PA_SUBSCRIPTION_EVENT_SAMPLE_CACHE|PA_SUBSCRIPTION_EVENT_CHANGE, e->index);
323 if (e->volume_is_set) {
324 if (pa_cvolume_valid(&e->volume))
325 pa_cvolume_remap(&e->volume, &old_channel_map, &e->channel_map);
326 else
327 pa_cvolume_reset(&e->volume, e->sample_spec.channels);
331 if (!e->memchunk.memblock)
332 goto fail;
334 pa_log_debug("Playing sample \"%s\" on \"%s\"", name, sink->name);
336 pass_volume = TRUE;
338 if (e->volume_is_set && PA_VOLUME_IS_VALID(volume)) {
339 pa_cvolume_set(&r, e->sample_spec.channels, volume);
340 pa_sw_cvolume_multiply(&r, &r, &e->volume);
341 } else if (e->volume_is_set)
342 r = e->volume;
343 else if (PA_VOLUME_IS_VALID(volume))
344 pa_cvolume_set(&r, e->sample_spec.channels, volume);
345 else
346 pass_volume = FALSE;
348 pa_proplist_update(merged, PA_UPDATE_REPLACE, e->proplist);
350 if (p)
351 pa_proplist_update(merged, PA_UPDATE_REPLACE, p);
353 if (pa_play_memchunk(sink,
354 &e->sample_spec, &e->channel_map,
355 &e->memchunk,
356 pass_volume ? &r : NULL,
357 merged,
358 PA_SINK_INPUT_NO_CREATE_ON_SUSPEND|PA_SINK_INPUT_KILL_ON_SUSPEND, sink_input_idx) < 0)
359 goto fail;
361 pa_proplist_free(merged);
363 if (e->lazy)
364 time(&e->last_used_time);
366 return 0;
368 fail:
369 pa_proplist_free(merged);
370 return -1;
373 int pa_scache_play_item_by_name(pa_core *c, const char *name, const char*sink_name, pa_volume_t volume, pa_proplist *p, uint32_t *sink_input_idx) {
374 pa_sink *sink;
376 pa_assert(c);
377 pa_assert(name);
379 if (!(sink = pa_namereg_get(c, sink_name, PA_NAMEREG_SINK)))
380 return -1;
382 return pa_scache_play_item(c, name, sink, volume, p, sink_input_idx);
385 const char *pa_scache_get_name_by_id(pa_core *c, uint32_t id) {
386 pa_scache_entry *e;
388 pa_assert(c);
389 pa_assert(id != PA_IDXSET_INVALID);
391 if (!c->scache || !(e = pa_idxset_get_by_index(c->scache, id)))
392 return NULL;
394 return e->name;
397 uint32_t pa_scache_get_id_by_name(pa_core *c, const char *name) {
398 pa_scache_entry *e;
400 pa_assert(c);
401 pa_assert(name);
403 if (!(e = pa_namereg_get(c, name, PA_NAMEREG_SAMPLE)))
404 return PA_IDXSET_INVALID;
406 return e->index;
409 size_t pa_scache_total_size(pa_core *c) {
410 pa_scache_entry *e;
411 uint32_t idx;
412 size_t sum = 0;
414 pa_assert(c);
416 if (!c->scache || !pa_idxset_size(c->scache))
417 return 0;
419 for (e = pa_idxset_first(c->scache, &idx); e; e = pa_idxset_next(c->scache, &idx))
420 if (e->memchunk.memblock)
421 sum += e->memchunk.length;
423 return sum;
426 void pa_scache_unload_unused(pa_core *c) {
427 pa_scache_entry *e;
428 time_t now;
429 uint32_t idx;
431 pa_assert(c);
433 if (!c->scache || !pa_idxset_size(c->scache))
434 return;
436 time(&now);
438 for (e = pa_idxset_first(c->scache, &idx); e; e = pa_idxset_next(c->scache, &idx)) {
440 if (!e->lazy || !e->memchunk.memblock)
441 continue;
443 if (e->last_used_time + c->scache_idle_time > now)
444 continue;
446 pa_memblock_unref(e->memchunk.memblock);
447 pa_memchunk_reset(&e->memchunk);
449 pa_subscription_post(c, PA_SUBSCRIPTION_EVENT_SAMPLE_CACHE|PA_SUBSCRIPTION_EVENT_CHANGE, e->index);
453 static void add_file(pa_core *c, const char *pathname) {
454 struct stat st;
455 const char *e;
457 pa_core_assert_ref(c);
458 pa_assert(pathname);
460 e = pa_path_get_filename(pathname);
462 if (stat(pathname, &st) < 0) {
463 pa_log("stat('%s'): %s", pathname, pa_cstrerror(errno));
464 return;
467 #if defined(S_ISREG) && defined(S_ISLNK)
468 if (S_ISREG(st.st_mode) || S_ISLNK(st.st_mode))
469 #endif
470 pa_scache_add_file_lazy(c, e, pathname, NULL);
473 int pa_scache_add_directory_lazy(pa_core *c, const char *pathname) {
474 DIR *dir;
476 pa_core_assert_ref(c);
477 pa_assert(pathname);
479 /* First try to open this as directory */
480 if (!(dir = opendir(pathname))) {
481 #ifdef HAVE_GLOB_H
482 glob_t p;
483 unsigned int i;
484 /* If that fails, try to open it as shell glob */
486 if (glob(pathname, GLOB_ERR|GLOB_NOSORT, NULL, &p) < 0) {
487 pa_log("failed to open directory '%s': %s", pathname, pa_cstrerror(errno));
488 return -1;
491 for (i = 0; i < p.gl_pathc; i++)
492 add_file(c, p.gl_pathv[i]);
494 globfree(&p);
495 #else
496 return -1;
497 #endif
498 } else {
499 struct dirent *e;
501 while ((e = readdir(dir))) {
502 char *p;
504 if (e->d_name[0] == '.')
505 continue;
507 p = pa_sprintf_malloc("%s" PA_PATH_SEP "%s", pathname, e->d_name);
508 add_file(c, p);
509 pa_xfree(p);
512 closedir(dir);
515 return 0;