scsi: fix tracing of scsi requests with simple backend
[qemu/aliguori.git] / audio / audio_pt_int.c
blob908c569a92031398d0e8150239c18fd47982f108
1 #include "qemu-common.h"
2 #include "audio.h"
4 #define AUDIO_CAP "audio-pt"
6 #include "audio_int.h"
7 #include "audio_pt_int.h"
9 #include <signal.h>
11 static void GCC_FMT_ATTR(3, 4) logerr (struct audio_pt *pt, int err,
12 const char *fmt, ...)
14 va_list ap;
16 va_start (ap, fmt);
17 AUD_vlog (pt->drv, fmt, ap);
18 va_end (ap);
20 AUD_log (NULL, "\n");
21 AUD_log (pt->drv, "Reason: %s\n", strerror (err));
24 int audio_pt_init (struct audio_pt *p, void *(*func) (void *),
25 void *opaque, const char *drv, const char *cap)
27 int err, err2;
28 const char *efunc;
29 sigset_t set, old_set;
31 p->drv = drv;
33 err = sigfillset (&set);
34 if (err) {
35 logerr (p, errno, "%s(%s): sigfillset failed", cap, AUDIO_FUNC);
36 return -1;
39 err = pthread_mutex_init (&p->mutex, NULL);
40 if (err) {
41 efunc = "pthread_mutex_init";
42 goto err0;
45 err = pthread_cond_init (&p->cond, NULL);
46 if (err) {
47 efunc = "pthread_cond_init";
48 goto err1;
51 err = pthread_sigmask (SIG_BLOCK, &set, &old_set);
52 if (err) {
53 efunc = "pthread_sigmask";
54 goto err2;
57 err = pthread_create (&p->thread, NULL, func, opaque);
59 err2 = pthread_sigmask (SIG_SETMASK, &old_set, NULL);
60 if (err2) {
61 logerr (p, err2, "%s(%s): pthread_sigmask (restore) failed",
62 cap, AUDIO_FUNC);
63 /* We have failed to restore original signal mask, all bets are off,
64 so terminate the process */
65 exit (EXIT_FAILURE);
68 if (err) {
69 efunc = "pthread_create";
70 goto err2;
73 return 0;
75 err2:
76 err2 = pthread_cond_destroy (&p->cond);
77 if (err2) {
78 logerr (p, err2, "%s(%s): pthread_cond_destroy failed", cap, AUDIO_FUNC);
81 err1:
82 err2 = pthread_mutex_destroy (&p->mutex);
83 if (err2) {
84 logerr (p, err2, "%s(%s): pthread_mutex_destroy failed", cap, AUDIO_FUNC);
87 err0:
88 logerr (p, err, "%s(%s): %s failed", cap, AUDIO_FUNC, efunc);
89 return -1;
92 int audio_pt_fini (struct audio_pt *p, const char *cap)
94 int err, ret = 0;
96 err = pthread_cond_destroy (&p->cond);
97 if (err) {
98 logerr (p, err, "%s(%s): pthread_cond_destroy failed", cap, AUDIO_FUNC);
99 ret = -1;
102 err = pthread_mutex_destroy (&p->mutex);
103 if (err) {
104 logerr (p, err, "%s(%s): pthread_mutex_destroy failed", cap, AUDIO_FUNC);
105 ret = -1;
107 return ret;
110 int audio_pt_lock (struct audio_pt *p, const char *cap)
112 int err;
114 err = pthread_mutex_lock (&p->mutex);
115 if (err) {
116 logerr (p, err, "%s(%s): pthread_mutex_lock failed", cap, AUDIO_FUNC);
117 return -1;
119 return 0;
122 int audio_pt_unlock (struct audio_pt *p, const char *cap)
124 int err;
126 err = pthread_mutex_unlock (&p->mutex);
127 if (err) {
128 logerr (p, err, "%s(%s): pthread_mutex_unlock failed", cap, AUDIO_FUNC);
129 return -1;
131 return 0;
134 int audio_pt_wait (struct audio_pt *p, const char *cap)
136 int err;
138 err = pthread_cond_wait (&p->cond, &p->mutex);
139 if (err) {
140 logerr (p, err, "%s(%s): pthread_cond_wait failed", cap, AUDIO_FUNC);
141 return -1;
143 return 0;
146 int audio_pt_unlock_and_signal (struct audio_pt *p, const char *cap)
148 int err;
150 err = pthread_mutex_unlock (&p->mutex);
151 if (err) {
152 logerr (p, err, "%s(%s): pthread_mutex_unlock failed", cap, AUDIO_FUNC);
153 return -1;
155 err = pthread_cond_signal (&p->cond);
156 if (err) {
157 logerr (p, err, "%s(%s): pthread_cond_signal failed", cap, AUDIO_FUNC);
158 return -1;
160 return 0;
163 int audio_pt_join (struct audio_pt *p, void **arg, const char *cap)
165 int err;
166 void *ret;
168 err = pthread_join (p->thread, &ret);
169 if (err) {
170 logerr (p, err, "%s(%s): pthread_join failed", cap, AUDIO_FUNC);
171 return -1;
173 *arg = ret;
174 return 0;