Commit 45334a23 authored by Eric Wong's avatar Eric Wong Committed by Max Kellermann

songvec: lock traversals for thread-safe updates/reads

Only one lock is used for all songvec traversals since they're rarely changed. Also, minimize lock time and release it before calling iterator functions since they may block (updateSongInfo => stat/open/seek/read). This lock only protects songvecs (and all of them) during traversals; not the individual song structures themselves.
parent bb04c634
#include "songvec.h" #include "songvec.h"
#include "utils.h" #include "utils.h"
static pthread_mutex_t nr_lock = PTHREAD_MUTEX_INITIALIZER;
/* Only used for sorting/searchin a songvec, not general purpose compares */ /* Only used for sorting/searchin a songvec, not general purpose compares */
static int songvec_cmp(const void *s1, const void *s2) static int songvec_cmp(const void *s1, const void *s2)
{ {
...@@ -16,23 +18,32 @@ static size_t sv_size(struct songvec *sv) ...@@ -16,23 +18,32 @@ static size_t sv_size(struct songvec *sv)
void songvec_sort(struct songvec *sv) void songvec_sort(struct songvec *sv)
{ {
pthread_mutex_lock(&nr_lock);
qsort(sv->base, sv->nr, sizeof(Song *), songvec_cmp); qsort(sv->base, sv->nr, sizeof(Song *), songvec_cmp);
pthread_mutex_unlock(&nr_lock);
} }
Song *songvec_find(struct songvec *sv, const char *url) Song *songvec_find(struct songvec *sv, const char *url)
{ {
int i; int i;
Song *ret = NULL;
for (i = sv->nr; --i >= 0; ) pthread_mutex_lock(&nr_lock);
if (!strcmp(sv->base[i]->url, url)) for (i = sv->nr; --i >= 0; ) {
return sv->base[i]; if (strcmp(sv->base[i]->url, url))
return NULL; continue;
ret = sv->base[i];
break;
}
pthread_mutex_unlock(&nr_lock);
return ret;
} }
int songvec_delete(struct songvec *sv, const Song *del) int songvec_delete(struct songvec *sv, const Song *del)
{ {
int i; int i;
pthread_mutex_lock(&nr_lock);
for (i = sv->nr; --i >= 0; ) { for (i = sv->nr; --i >= 0; ) {
if (sv->base[i] != del) if (sv->base[i] != del)
continue; continue;
...@@ -45,37 +56,50 @@ int songvec_delete(struct songvec *sv, const Song *del) ...@@ -45,37 +56,50 @@ int songvec_delete(struct songvec *sv, const Song *del)
(sv->nr - i + 1) * sizeof(Song *)); (sv->nr - i + 1) * sizeof(Song *));
sv->base = xrealloc(sv->base, sv_size(sv)); sv->base = xrealloc(sv->base, sv_size(sv));
} }
return i; break;
} }
pthread_mutex_unlock(&nr_lock);
return -1; /* not found */ return i;
} }
void songvec_add(struct songvec *sv, Song *add) void songvec_add(struct songvec *sv, Song *add)
{ {
pthread_mutex_lock(&nr_lock);
++sv->nr; ++sv->nr;
sv->base = xrealloc(sv->base, sv_size(sv)); sv->base = xrealloc(sv->base, sv_size(sv));
sv->base[sv->nr - 1] = add; sv->base[sv->nr - 1] = add;
pthread_mutex_unlock(&nr_lock);
} }
void songvec_destroy(struct songvec *sv) void songvec_destroy(struct songvec *sv)
{ {
pthread_mutex_lock(&nr_lock);
if (sv->base) { if (sv->base) {
free(sv->base); free(sv->base);
sv->base = NULL; sv->base = NULL;
} }
sv->nr = 0; sv->nr = 0;
pthread_mutex_unlock(&nr_lock);
} }
int songvec_for_each(struct songvec *sv, int (*fn)(Song *, void *), void *arg) int songvec_for_each(struct songvec *sv, int (*fn)(Song *, void *), void *arg)
{ {
int i; size_t i;
Song **sp = sv->base;
for (i = sv->nr; --i >= 0; ) { pthread_mutex_lock(&nr_lock);
if (fn(*sp++, arg) < 0) for (i = 0; i < sv->nr; ++i) {
Song *song = sv->base[i];
assert(song);
assert(*song->url);
pthread_mutex_unlock(&nr_lock); /* fn() may block */
if (fn(song, arg) < 0)
return -1; return -1;
pthread_mutex_lock(&nr_lock); /* sv->nr may change in fn() */
} }
pthread_mutex_unlock(&nr_lock);
return 0; return 0;
} }
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment