Mercurial > audlegacy-plugins
annotate src/musepack/libmpc.cxx @ 316:fb513e10174e trunk
[svn] - merge libconsole-blargg into mainline libconsole:
+ obsoletes plugins-ugly:sapplug
author | nenolod |
---|---|
date | Thu, 30 Nov 2006 19:54:33 -0800 |
parents | 4f7b72c88319 |
children | c812e846b84e |
rev | line source |
---|---|
232 | 1 #include "libmpc.h" |
2 | |
3 #define FORCED_THREAD_STACKSIZE 1024 * 1000 | |
4 #define REMOVE_NONEXISTANT_TAG(x) if (!*x) { x = NULL; } | |
5 | |
6 using TagLib::MPC::File; | |
7 using TagLib::Tag; | |
8 using TagLib::String; | |
9 using TagLib::APE::ItemListMap; | |
10 | |
11 InputPlugin MpcPlugin = { | |
12 NULL, //File Handle FILE* handle | |
13 NULL, //Filename char* filename | |
14 NULL, //Name of Plugin char* filename | |
15 mpcOpenPlugin, //Open Plugin [CALLBACK] | |
16 mpcAboutBox, //Show About box [CALLBACK] | |
17 mpcConfigBox, //Show Configure box [CALLBACK] | |
260
4f7b72c88319
[svn] So input.c wants to have the old-style function available...
chainsaw
parents:
254
diff
changeset
|
18 mpcIsOurFile, //Check if it's our file [CALLBACK] |
232 | 19 NULL, //Scan the directory [UNUSED] |
20 mpcPlay, //Play [CALLBACK] | |
21 mpcStop, //Stop [CALLBACK] | |
22 mpcPause, //Pause [CALLBACK] | |
23 mpcSeek, //Seek [CALLBACK] | |
24 NULL, //Set EQ [CALLBACK] | |
25 mpcGetTime, //Get Time [CALLBACK] | |
26 NULL, //Get Volume [UNUSED] | |
27 NULL, //Set Volume [UNUSED] | |
28 NULL, //Close Plugin [UNUSED] | |
29 NULL, //Obsolete [UNUSED] | |
30 NULL, //Visual plugins add_vis_pcm(int time, AFormat fmt, int nch, int length, void *ptr) | |
31 NULL, //Set Info Settings set_info(char *title, int length, int rate, int freq, int nch) | |
32 NULL, //set Info Text set_info_text(char* text) | |
33 mpcGetSongInfo, //Get Title String callback [CALLBACK] | |
34 mpcFileInfoBox, //Show File Info Box [CALLBACK] | |
35 NULL, //Output Plugin Handle OutputPlugin output | |
254 | 36 mpcGetSongTuple,//Acquire tuple for song [CALLBACK] |
37 NULL, | |
38 NULL, | |
39 mpcIsOurFD, | |
232 | 40 }; |
41 | |
42 extern "C" | |
43 InputPlugin* get_iplugin_info() | |
44 { | |
45 MpcPlugin.description = g_strdup_printf("Musepack Audio Plugin"); | |
46 return &MpcPlugin; | |
47 } | |
48 | |
49 static PluginConfig pluginConfig = {0}; | |
50 static Widgets widgets = {0}; | |
51 static MpcDecoder mpcDecoder = {0}; | |
52 static TrackInfo track = {0}; | |
53 | |
54 static GThread *threadHandle; | |
55 GStaticMutex threadMutex = G_STATIC_MUTEX_INIT; | |
56 | |
57 /* | |
58 * VFS callback implementation, adapted from mpc_reader.c. | |
59 * This _IS_ very sick, but it works. -nenolod | |
60 */ | |
61 static mpc_int32_t | |
62 vfs_fread_impl(void *data, void *ptr, mpc_int32_t size) | |
63 { | |
64 mpc_reader_file *d = (mpc_reader_file *) data; | |
65 VFSFile *file = (VFSFile *) d->file; | |
66 | |
67 return (mpc_int32_t) vfs_fread(ptr, 1, size, file); | |
68 } | |
69 | |
70 static mpc_bool_t | |
71 vfs_fseek_impl(void *data, mpc_int32_t offset) | |
72 { | |
73 mpc_reader_file *d = (mpc_reader_file *) data; | |
74 VFSFile *file = (VFSFile *) d->file; | |
75 | |
76 return d->is_seekable ? vfs_fseek(file, offset, SEEK_SET) == 0 : FALSE; | |
77 } | |
78 | |
79 static mpc_int32_t | |
80 vfs_ftell_impl(void *data) | |
81 { | |
82 mpc_reader_file *d = (mpc_reader_file *) data; | |
83 VFSFile *file = (VFSFile *) d->file; | |
84 | |
85 return vfs_ftell(file); | |
86 } | |
87 | |
88 static mpc_int32_t | |
89 vfs_getsize_impl(void *data) | |
90 { | |
91 mpc_reader_file *d = (mpc_reader_file *) data; | |
92 | |
93 return d->file_size; | |
94 } | |
95 | |
96 static mpc_bool_t | |
97 vfs_canseek_impl(void *data) | |
98 { | |
99 mpc_reader_file *d = (mpc_reader_file *) data; | |
100 | |
101 return d->is_seekable; | |
102 } | |
103 | |
104 /* | |
105 * This sets up an mpc_reader_file object to read from VFS instead of libc. | |
106 * Essentially, we use this instead of the normal constructor. | |
107 * - nenolod | |
108 */ | |
109 void | |
110 mpc_reader_setup_file_vfs(mpc_reader_file *p_reader, VFSFile *input) | |
111 { | |
112 p_reader->reader.seek = vfs_fseek_impl; | |
113 p_reader->reader.read = vfs_fread_impl; | |
114 p_reader->reader.tell = vfs_ftell_impl; | |
115 p_reader->reader.get_size = vfs_getsize_impl; | |
116 p_reader->reader.canseek = vfs_canseek_impl; | |
117 p_reader->reader.data = p_reader; | |
118 | |
119 p_reader->file = (FILE *) input; // no worries, it gets cast back -nenolod | |
120 p_reader->is_seekable = TRUE; // XXX streams | |
121 | |
122 vfs_fseek(input, 0, SEEK_END); | |
123 p_reader->file_size = vfs_ftell(input); | |
124 vfs_fseek(input, 0, SEEK_SET); | |
125 } | |
126 | |
127 static void mpcOpenPlugin() | |
128 { | |
129 ConfigDb *cfg; | |
130 cfg = bmp_cfg_db_open(); | |
131 bmp_cfg_db_get_bool(cfg, "musepack", "clipPrevention", &pluginConfig.clipPrevention); | |
132 bmp_cfg_db_get_bool(cfg, "musepack", "albumGain", &pluginConfig.albumGain); | |
133 bmp_cfg_db_get_bool(cfg, "musepack", "dynamicBitrate", &pluginConfig.dynamicBitrate); | |
134 bmp_cfg_db_get_bool(cfg, "musepack", "replaygain", &pluginConfig.replaygain); | |
135 bmp_cfg_db_close(cfg); | |
136 } | |
137 | |
138 static void mpcAboutBox() | |
139 { | |
140 GtkWidget* aboutBox = widgets.aboutBox; | |
141 if (aboutBox) | |
142 gdk_window_raise(aboutBox->window); | |
143 else | |
144 { | |
145 char* titleText = g_strdup_printf("Musepack Decoder Plugin 1.2"); | |
146 char* contentText = "Plugin code by\nBenoit Amiaux\nMartin Spuler\nKuniklo\n\nGet latest version at http://musepack.net\n"; | |
147 char* buttonText = "Nevermind"; | |
148 aboutBox = xmms_show_message(titleText, contentText, buttonText, FALSE, NULL, NULL); | |
149 widgets.aboutBox = aboutBox; | |
150 g_signal_connect(G_OBJECT(aboutBox), "destroy", G_CALLBACK(gtk_widget_destroyed), &widgets.aboutBox); | |
151 } | |
152 } | |
153 | |
154 static void mpcConfigBox() | |
155 { | |
156 GtkWidget* configBox = widgets.configBox; | |
157 if(configBox) | |
158 gdk_window_raise(configBox->window); | |
159 else | |
160 { | |
161 configBox = gtk_window_new(GTK_WINDOW_TOPLEVEL); | |
162 gtk_window_set_type_hint(GTK_WINDOW(configBox), GDK_WINDOW_TYPE_HINT_DIALOG); | |
163 widgets.configBox = configBox; | |
164 g_signal_connect(G_OBJECT(configBox), "destroy", G_CALLBACK(gtk_widget_destroyed), &widgets.configBox); | |
165 gtk_window_set_title(GTK_WINDOW(configBox), "Musepack Decoder Configuration"); | |
166 gtk_window_set_policy(GTK_WINDOW(configBox), FALSE, FALSE, FALSE); | |
167 gtk_container_border_width(GTK_CONTAINER(configBox), 10); | |
168 | |
169 GtkWidget* notebook = gtk_notebook_new(); | |
170 GtkWidget* vbox = gtk_vbox_new(FALSE, 10); | |
171 gtk_box_pack_start(GTK_BOX(vbox), notebook, TRUE, TRUE, 0); | |
172 gtk_container_add(GTK_CONTAINER(configBox), vbox); | |
173 | |
174 //General Settings Tab | |
175 GtkWidget* generalSet = gtk_frame_new("General Settings"); | |
176 gtk_container_border_width(GTK_CONTAINER(generalSet), 5); | |
177 | |
178 GtkWidget* gSvbox = gtk_vbox_new(FALSE, 10); | |
179 gtk_container_border_width(GTK_CONTAINER(gSvbox), 5); | |
180 gtk_container_add(GTK_CONTAINER(generalSet), gSvbox); | |
181 | |
182 GtkWidget* bitrateCheck = gtk_check_button_new_with_label("Enable Dynamic Bitrate Display"); | |
183 widgets.bitrateCheck = bitrateCheck; | |
184 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(bitrateCheck), pluginConfig.dynamicBitrate); | |
185 gtk_box_pack_start(GTK_BOX(gSvbox), bitrateCheck, FALSE, FALSE, 0); | |
186 gtk_notebook_append_page(GTK_NOTEBOOK(notebook), generalSet, gtk_label_new("Plugin")); | |
187 | |
188 //ReplayGain Settings Tab | |
189 GtkWidget* replaygainSet = gtk_frame_new("ReplayGain Settings"); | |
190 gtk_container_border_width(GTK_CONTAINER(replaygainSet), 5); | |
191 | |
192 GtkWidget* rSVbox = gtk_vbox_new(FALSE, 10); | |
193 gtk_container_border_width(GTK_CONTAINER(rSVbox), 5); | |
194 gtk_container_add(GTK_CONTAINER(replaygainSet), rSVbox); | |
195 | |
196 GtkWidget* clippingCheck = gtk_check_button_new_with_label("Enable Clipping Prevention"); | |
197 widgets.clippingCheck = clippingCheck; | |
198 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(clippingCheck), pluginConfig.clipPrevention); | |
199 gtk_box_pack_start(GTK_BOX(rSVbox), clippingCheck, FALSE, FALSE, 0); | |
200 | |
201 GtkWidget* replaygainCheck = gtk_check_button_new_with_label("Enable ReplayGain"); | |
202 widgets.replaygainCheck = replaygainCheck; | |
203 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(replaygainCheck), pluginConfig.replaygain); | |
204 gtk_box_pack_start(GTK_BOX(rSVbox), replaygainCheck, FALSE, FALSE, 0); | |
205 | |
206 GtkWidget* replaygainType = gtk_frame_new("ReplayGain Type"); | |
207 gtk_box_pack_start(GTK_BOX(rSVbox), replaygainType, FALSE, FALSE, 0); | |
208 g_signal_connect(G_OBJECT(replaygainCheck), "toggled", G_CALLBACK(toggleSwitch), replaygainType); | |
209 | |
210 GtkWidget* rgVbox = gtk_vbox_new(FALSE, 5); | |
211 gtk_container_set_border_width(GTK_CONTAINER(rgVbox), 5); | |
212 gtk_container_add(GTK_CONTAINER(replaygainType), rgVbox); | |
213 | |
214 GtkWidget* trackCheck = gtk_radio_button_new_with_label(NULL, "Use Track Gain"); | |
215 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(trackCheck), !pluginConfig.albumGain); | |
216 gtk_box_pack_start(GTK_BOX(rgVbox), trackCheck, FALSE, FALSE, 0); | |
217 | |
218 GtkWidget* albumCheck = gtk_radio_button_new_with_label(gtk_radio_button_group(GTK_RADIO_BUTTON(trackCheck)), "Use Album Gain"); | |
219 widgets.albumCheck = albumCheck; | |
220 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(albumCheck), pluginConfig.albumGain); | |
221 gtk_box_pack_start(GTK_BOX(rgVbox), albumCheck, FALSE, FALSE, 0); | |
222 gtk_widget_set_sensitive(replaygainType, pluginConfig.replaygain); | |
223 gtk_notebook_append_page(GTK_NOTEBOOK(notebook), replaygainSet, gtk_label_new("ReplayGain")); | |
224 | |
225 //Buttons | |
226 GtkWidget* buttonBox = gtk_hbutton_box_new(); | |
227 gtk_button_box_set_layout(GTK_BUTTON_BOX(buttonBox), GTK_BUTTONBOX_END); | |
228 gtk_button_box_set_spacing(GTK_BUTTON_BOX(buttonBox), 5); | |
229 gtk_box_pack_start(GTK_BOX(vbox), buttonBox, FALSE, FALSE, 0); | |
230 | |
231 GtkWidget* okButton = gtk_button_new_with_label("Ok"); | |
232 g_signal_connect(G_OBJECT(okButton), "clicked", G_CALLBACK(saveConfigBox), NULL); | |
233 GTK_WIDGET_SET_FLAGS(okButton, GTK_CAN_DEFAULT); | |
234 gtk_box_pack_start(GTK_BOX(buttonBox), okButton, TRUE, TRUE, 0); | |
235 | |
236 GtkWidget* cancelButton = gtk_button_new_with_label("Cancel"); | |
237 g_signal_connect_swapped(G_OBJECT(cancelButton), "clicked", G_CALLBACK(gtk_widget_destroy), GTK_OBJECT(widgets.configBox)); | |
238 GTK_WIDGET_SET_FLAGS(cancelButton, GTK_CAN_DEFAULT); | |
239 gtk_widget_grab_default(cancelButton); | |
240 gtk_box_pack_start(GTK_BOX(buttonBox), cancelButton, TRUE, TRUE, 0); | |
241 | |
242 gtk_widget_show_all(configBox); | |
243 } | |
244 } | |
245 | |
246 static void toggleSwitch(GtkWidget* p_Widget, gpointer p_Data) | |
247 { | |
248 gtk_widget_set_sensitive(GTK_WIDGET(p_Data), gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(p_Widget))); | |
249 } | |
250 | |
251 static void saveConfigBox(GtkWidget* p_Widget, gpointer p_Data) | |
252 { | |
253 ConfigDb* cfg; | |
254 GtkToggleButton* tb; | |
255 | |
256 tb = GTK_TOGGLE_BUTTON(widgets.replaygainCheck); | |
257 pluginConfig.replaygain = gtk_toggle_button_get_active(tb); | |
258 tb = GTK_TOGGLE_BUTTON(widgets.clippingCheck); | |
259 pluginConfig.clipPrevention = gtk_toggle_button_get_active(tb); | |
260 tb = GTK_TOGGLE_BUTTON(widgets.bitrateCheck); | |
261 pluginConfig.dynamicBitrate = gtk_toggle_button_get_active(tb); | |
262 tb = GTK_TOGGLE_BUTTON(widgets.albumCheck); | |
263 pluginConfig.albumGain = gtk_toggle_button_get_active(tb); | |
264 | |
265 cfg = bmp_cfg_db_open(); | |
266 | |
267 bmp_cfg_db_set_bool(cfg, "musepack", "clipPrevention", pluginConfig.clipPrevention); | |
268 bmp_cfg_db_set_bool(cfg, "musepack", "albumGain", pluginConfig.albumGain); | |
269 bmp_cfg_db_set_bool(cfg, "musepack", "dynamicBitrate", pluginConfig.dynamicBitrate); | |
270 bmp_cfg_db_set_bool(cfg, "musepack", "replaygain", pluginConfig.replaygain); | |
271 | |
272 bmp_cfg_db_close(cfg); | |
273 | |
274 gtk_widget_destroy (widgets.configBox); | |
275 } | |
276 | |
260
4f7b72c88319
[svn] So input.c wants to have the old-style function available...
chainsaw
parents:
254
diff
changeset
|
277 static int mpcIsOurFile(char* p_Filename) |
4f7b72c88319
[svn] So input.c wants to have the old-style function available...
chainsaw
parents:
254
diff
changeset
|
278 { |
4f7b72c88319
[svn] So input.c wants to have the old-style function available...
chainsaw
parents:
254
diff
changeset
|
279 VFSFile *file; |
4f7b72c88319
[svn] So input.c wants to have the old-style function available...
chainsaw
parents:
254
diff
changeset
|
280 gchar magic[3]; |
4f7b72c88319
[svn] So input.c wants to have the old-style function available...
chainsaw
parents:
254
diff
changeset
|
281 if ((file = vfs_fopen(p_Filename, "rb"))) { |
4f7b72c88319
[svn] So input.c wants to have the old-style function available...
chainsaw
parents:
254
diff
changeset
|
282 vfs_fread(magic, 1, 3, file); |
4f7b72c88319
[svn] So input.c wants to have the old-style function available...
chainsaw
parents:
254
diff
changeset
|
283 if (!memcmp(magic, "MP+", 3)) { |
4f7b72c88319
[svn] So input.c wants to have the old-style function available...
chainsaw
parents:
254
diff
changeset
|
284 vfs_fclose(file); |
4f7b72c88319
[svn] So input.c wants to have the old-style function available...
chainsaw
parents:
254
diff
changeset
|
285 return 1; |
4f7b72c88319
[svn] So input.c wants to have the old-style function available...
chainsaw
parents:
254
diff
changeset
|
286 } |
4f7b72c88319
[svn] So input.c wants to have the old-style function available...
chainsaw
parents:
254
diff
changeset
|
287 vfs_fclose(file); |
4f7b72c88319
[svn] So input.c wants to have the old-style function available...
chainsaw
parents:
254
diff
changeset
|
288 } |
4f7b72c88319
[svn] So input.c wants to have the old-style function available...
chainsaw
parents:
254
diff
changeset
|
289 return 0; |
4f7b72c88319
[svn] So input.c wants to have the old-style function available...
chainsaw
parents:
254
diff
changeset
|
290 } |
4f7b72c88319
[svn] So input.c wants to have the old-style function available...
chainsaw
parents:
254
diff
changeset
|
291 |
254 | 292 static int mpcIsOurFD(char* p_Filename, VFSFile* file) |
232 | 293 { |
254 | 294 gchar magic[3]; |
295 vfs_fread(magic, 1, 3, file); | |
260
4f7b72c88319
[svn] So input.c wants to have the old-style function available...
chainsaw
parents:
254
diff
changeset
|
296 if (!memcmp(magic, "MP+", 3)) |
254 | 297 return 1; |
298 return 0; | |
232 | 299 } |
300 | |
301 static void mpcPlay(char* p_Filename) | |
302 { | |
303 mpcDecoder.offset = -1; | |
304 mpcDecoder.isAlive = true; | |
305 mpcDecoder.isOutput = false; | |
306 mpcDecoder.isPause = false; | |
307 threadHandle = g_thread_create(GThreadFunc(decodeStream), (void *) g_strdup(p_Filename), TRUE, NULL); | |
308 } | |
309 | |
310 static void mpcStop() | |
311 { | |
312 setAlive(false); | |
313 if (threadHandle) | |
314 { | |
315 g_thread_join(threadHandle); | |
316 if (mpcDecoder.isOutput) | |
317 { | |
318 MpcPlugin.output->buffer_free(); | |
319 MpcPlugin.output->close_audio(); | |
320 mpcDecoder.isOutput = false; | |
321 } | |
322 } | |
323 } | |
324 | |
325 static void mpcPause(short p_Pause) | |
326 { | |
327 lockAcquire(); | |
328 mpcDecoder.isPause = p_Pause; | |
329 MpcPlugin.output->pause(p_Pause); | |
330 lockRelease(); | |
331 } | |
332 | |
333 static void mpcSeek(int p_Offset) | |
334 { | |
335 lockAcquire(); | |
336 mpcDecoder.offset = static_cast<double> (p_Offset); | |
337 MpcPlugin.output->flush(1000 * p_Offset); | |
338 lockRelease(); | |
339 } | |
340 | |
341 static int mpcGetTime() | |
342 { | |
343 if(!isAlive()) | |
344 return -1; | |
345 return MpcPlugin.output->output_time(); | |
346 } | |
347 | |
348 static TitleInput *mpcGetSongTuple(char* p_Filename) | |
349 { | |
350 VFSFile *input = vfs_fopen(p_Filename, "rb"); | |
351 TitleInput *tuple = NULL; | |
352 | |
353 if(input) | |
354 { | |
355 tuple = bmp_title_input_new(); | |
356 gchar *filename_proxy = g_strdup(p_Filename); | |
357 | |
358 tuple->file_name = g_path_get_basename(filename_proxy); | |
359 tuple->file_path = g_path_get_dirname(filename_proxy); | |
360 tuple->file_ext = "mpc"; // XXX: I can't be assed. -nenolod | |
361 | |
362 MpcInfo tags = getTags(p_Filename); | |
363 | |
364 tuple->date = g_strdup(tags.date); | |
365 tuple->track_name = g_strdup(tags.title); | |
366 tuple->performer = g_strdup(tags.artist); | |
367 tuple->album_name = g_strdup(tags.album); | |
368 tuple->track_number = tags.track; | |
369 tuple->year = tags.year; | |
370 tuple->genre = g_strdup(tags.genre); | |
371 tuple->comment = g_strdup(tags.comment); | |
372 | |
373 freeTags(tags); | |
374 | |
375 mpc_streaminfo info; | |
376 mpc_reader_file reader; | |
377 mpc_reader_setup_file_vfs(&reader, input); | |
378 mpc_streaminfo_read(&info, &reader.reader); | |
379 | |
380 tuple->length = static_cast<int> (1000 * mpc_streaminfo_get_length(&info)); | |
381 vfs_fclose(input); | |
382 } | |
383 else | |
384 { | |
385 char* temp = g_strdup_printf("[xmms-musepack] mpcGetSongInfo is unable to open %s\n", p_Filename); | |
386 perror(temp); | |
387 free(temp); | |
388 } | |
389 | |
390 return tuple; | |
391 } | |
392 | |
393 static void mpcGetSongInfo(char* p_Filename, char** p_Title, int* p_Length) | |
394 { | |
395 VFSFile *input = vfs_fopen(p_Filename, "rb"); | |
396 if(input) | |
397 { | |
398 MpcInfo tags = getTags(p_Filename); | |
399 *p_Title = mpcGenerateTitle(tags, p_Filename); | |
400 freeTags(tags); | |
401 mpc_streaminfo info; | |
402 mpc_reader_file reader; | |
403 mpc_reader_setup_file_vfs(&reader, input); | |
404 mpc_streaminfo_read(&info, &reader.reader); | |
405 *p_Length = static_cast<int> (1000 * mpc_streaminfo_get_length(&info)); | |
406 vfs_fclose(input); | |
407 } | |
408 else | |
409 { | |
410 char* temp = g_strdup_printf("[xmms-musepack] mpcGetSongInfo is unable to open %s\n", p_Filename); | |
411 perror(temp); | |
412 free(temp); | |
413 } | |
414 } | |
415 | |
416 static void freeTags(MpcInfo& tags) | |
417 { | |
418 free(tags.title); | |
419 free(tags.artist); | |
420 free(tags.album); | |
421 free(tags.comment); | |
422 free(tags.genre); | |
423 free(tags.date); | |
424 } | |
425 | |
426 static MpcInfo getTags(const char* p_Filename) | |
427 { | |
428 File oFile(p_Filename, false); | |
429 Tag* poTag = oFile.tag(); | |
430 MpcInfo tags = {0}; | |
431 tags.title = g_strdup(poTag->title().toCString(true)); | |
432 REMOVE_NONEXISTANT_TAG(tags.title); | |
433 tags.artist = g_strdup(poTag->artist().toCString(true)); | |
434 REMOVE_NONEXISTANT_TAG(tags.artist); | |
435 tags.album = g_strdup(poTag->album().toCString(true)); | |
436 REMOVE_NONEXISTANT_TAG(tags.album); | |
437 tags.genre = g_strdup(poTag->genre().toCString(true)); | |
438 REMOVE_NONEXISTANT_TAG(tags.genre); | |
439 tags.comment = g_strdup(poTag->comment().toCString(true)); | |
440 REMOVE_NONEXISTANT_TAG(tags.comment); | |
441 tags.year = poTag->year(); | |
442 tags.track = poTag->track(); | |
443 TagLib::APE::Tag* ape = oFile.APETag(false); | |
444 if(ape) | |
445 { | |
446 ItemListMap map = ape->itemListMap(); | |
447 if(map.contains("YEAR")) | |
448 { | |
449 tags.date = g_strdup(map["YEAR"].toString().toCString(true)); | |
450 } | |
451 else | |
452 { | |
453 tags.date = g_strdup_printf("%d", tags.year); | |
454 } | |
455 } | |
456 return tags; | |
457 } | |
458 | |
459 static void mpcFileInfoBox(char* p_Filename) | |
460 { | |
461 GtkWidget* infoBox = widgets.infoBox; | |
462 | |
463 if(infoBox) | |
464 gdk_window_raise(infoBox->window); | |
465 else | |
466 { | |
467 infoBox = gtk_window_new(GTK_WINDOW_TOPLEVEL); | |
468 gtk_window_set_type_hint(GTK_WINDOW(infoBox), GDK_WINDOW_TYPE_HINT_DIALOG); | |
469 widgets.infoBox = infoBox; | |
470 gtk_window_set_policy(GTK_WINDOW(infoBox), FALSE, FALSE, FALSE); | |
471 g_signal_connect(G_OBJECT(infoBox), "destroy", G_CALLBACK(closeInfoBox), NULL); | |
472 gtk_container_set_border_width(GTK_CONTAINER(infoBox), 10); | |
473 | |
474 GtkWidget* iVbox = gtk_vbox_new(FALSE, 10); | |
475 gtk_container_add(GTK_CONTAINER(infoBox), iVbox); | |
476 | |
477 GtkWidget* filenameHbox = gtk_hbox_new(FALSE, 5); | |
478 gtk_box_pack_start(GTK_BOX(iVbox), filenameHbox, FALSE, TRUE, 0); | |
479 | |
480 GtkWidget* fileLabel = gtk_label_new("Filename:"); | |
481 gtk_box_pack_start(GTK_BOX(filenameHbox), fileLabel, FALSE, TRUE, 0); | |
482 | |
483 GtkWidget* fileEntry = gtk_entry_new(); | |
484 widgets.fileEntry = fileEntry; | |
485 gtk_editable_set_editable(GTK_EDITABLE(fileEntry), FALSE); | |
486 gtk_box_pack_start(GTK_BOX(filenameHbox), fileEntry, TRUE, TRUE, 0); | |
487 | |
488 GtkWidget* iHbox = gtk_hbox_new(FALSE, 10); | |
489 gtk_box_pack_start(GTK_BOX(iVbox), iHbox, FALSE, TRUE, 0); | |
490 | |
491 GtkWidget* leftBox = gtk_vbox_new(FALSE, 10); | |
492 gtk_box_pack_start(GTK_BOX(iHbox), leftBox, FALSE, FALSE, 0); | |
493 | |
494 //Tag labels | |
495 GtkWidget* tagFrame = gtk_frame_new("Musepack Tag"); | |
496 gtk_box_pack_start(GTK_BOX(leftBox), tagFrame, FALSE, FALSE, 0); | |
497 gtk_widget_set_sensitive(tagFrame, TRUE); | |
498 | |
499 GtkWidget* iTable = gtk_table_new(5, 5, FALSE); | |
500 gtk_container_set_border_width(GTK_CONTAINER(iTable), 5); | |
501 gtk_container_add(GTK_CONTAINER(tagFrame), iTable); | |
502 | |
503 mpcGtkTagLabel("Title:", 0, 1, 0, 1, iTable); | |
504 GtkWidget* titleEntry = mpcGtkTagEntry(1, 4, 0, 1, 0, iTable); | |
505 widgets.titleEntry = titleEntry; | |
506 | |
507 mpcGtkTagLabel("Artist:", 0, 1, 1, 2, iTable); | |
508 GtkWidget* artistEntry = mpcGtkTagEntry(1, 4, 1, 2, 0, iTable); | |
509 widgets.artistEntry = artistEntry; | |
510 | |
511 mpcGtkTagLabel("Album:", 0, 1, 2, 3, iTable); | |
512 GtkWidget* albumEntry = mpcGtkTagEntry(1, 4, 2, 3, 0, iTable); | |
513 widgets.albumEntry = albumEntry; | |
514 | |
515 mpcGtkTagLabel("Comment:", 0, 1, 3, 4, iTable); | |
516 GtkWidget* commentEntry = mpcGtkTagEntry(1, 4, 3, 4, 0, iTable); | |
517 widgets.commentEntry = commentEntry; | |
518 | |
519 mpcGtkTagLabel("Year:", 0, 1, 4, 5, iTable); | |
520 GtkWidget* yearEntry = mpcGtkTagEntry(1, 2, 4, 5, 4, iTable); | |
521 widgets.yearEntry = yearEntry; | |
522 gtk_widget_set_usize(yearEntry, 4, -1); | |
523 | |
524 mpcGtkTagLabel("Track:", 2, 3, 4, 5, iTable); | |
525 GtkWidget* trackEntry = mpcGtkTagEntry(3, 4, 4, 5, 4, iTable); | |
526 widgets.trackEntry = trackEntry; | |
527 gtk_widget_set_usize(trackEntry, 3, -1); | |
528 | |
529 mpcGtkTagLabel("Genre:", 0, 1, 5, 6, iTable); | |
530 GtkWidget* genreEntry = mpcGtkTagEntry(1, 4, 5, 6, 0, iTable); | |
531 widgets.genreEntry = genreEntry; | |
532 gtk_widget_set_usize(genreEntry, 20, -1); | |
533 | |
534 //Buttons | |
535 GtkWidget* buttonBox = gtk_hbutton_box_new(); | |
536 gtk_button_box_set_layout(GTK_BUTTON_BOX(buttonBox), GTK_BUTTONBOX_END); | |
537 gtk_button_box_set_spacing(GTK_BUTTON_BOX(buttonBox), 5); | |
538 gtk_box_pack_start(GTK_BOX(leftBox), buttonBox, FALSE, FALSE, 0); | |
539 | |
540 GtkWidget* saveButton = mpcGtkButton("Save", buttonBox); | |
541 g_signal_connect(G_OBJECT(saveButton), "clicked", G_CALLBACK(saveTags), NULL); | |
542 | |
543 GtkWidget* removeButton = mpcGtkButton("Remove Tag", buttonBox); | |
544 g_signal_connect_swapped(G_OBJECT(removeButton), "clicked", G_CALLBACK(removeTags), NULL); | |
545 | |
546 GtkWidget* cancelButton = mpcGtkButton("Cancel", buttonBox); | |
547 g_signal_connect_swapped(G_OBJECT(cancelButton), "clicked", G_CALLBACK(closeInfoBox), NULL); | |
548 gtk_widget_grab_default(cancelButton); | |
549 | |
550 //File information | |
551 GtkWidget* infoFrame = gtk_frame_new("Musepack Info"); | |
552 gtk_box_pack_start(GTK_BOX(iHbox), infoFrame, FALSE, FALSE, 0); | |
553 | |
554 GtkWidget* infoVbox = gtk_vbox_new(FALSE, 5); | |
555 gtk_container_add(GTK_CONTAINER(infoFrame), infoVbox); | |
556 gtk_container_set_border_width(GTK_CONTAINER(infoVbox), 10); | |
557 gtk_box_set_spacing(GTK_BOX(infoVbox), 0); | |
558 | |
559 GtkWidget* streamLabel = mpcGtkLabel(infoVbox); | |
560 GtkWidget* encoderLabel = mpcGtkLabel(infoVbox); | |
561 GtkWidget* profileLabel = mpcGtkLabel(infoVbox); | |
562 GtkWidget* bitrateLabel = mpcGtkLabel(infoVbox); | |
563 GtkWidget* rateLabel = mpcGtkLabel(infoVbox); | |
564 GtkWidget* channelsLabel = mpcGtkLabel(infoVbox); | |
565 GtkWidget* lengthLabel = mpcGtkLabel(infoVbox); | |
566 GtkWidget* fileSizeLabel = mpcGtkLabel(infoVbox); | |
567 GtkWidget* trackPeakLabel = mpcGtkLabel(infoVbox); | |
568 GtkWidget* trackGainLabel = mpcGtkLabel(infoVbox); | |
569 GtkWidget* albumPeakLabel = mpcGtkLabel(infoVbox); | |
570 GtkWidget* albumGainLabel = mpcGtkLabel(infoVbox); | |
571 | |
572 VFSFile *input = vfs_fopen(p_Filename, "rb"); | |
573 if(input) | |
574 { | |
575 mpc_streaminfo info; | |
576 mpc_reader_file reader; | |
577 mpc_reader_setup_file_vfs(&reader, input); | |
578 mpc_streaminfo_read(&info, &reader.reader); | |
579 | |
580 int time = static_cast<int> (mpc_streaminfo_get_length(&info)); | |
581 int minutes = time / 60; | |
582 int seconds = time % 60; | |
583 | |
584 mpcGtkPrintLabel(streamLabel, "Streamversion %d", info.stream_version); | |
585 mpcGtkPrintLabel(encoderLabel, "Encoder: \%s", info.encoder); | |
586 mpcGtkPrintLabel(profileLabel, "Profile: \%s", info.profile_name); | |
587 mpcGtkPrintLabel(bitrateLabel, "Average bitrate: \%6.1f kbps", info.average_bitrate * 1.e-3); | |
588 mpcGtkPrintLabel(rateLabel, "Samplerate: \%d Hz", info.sample_freq); | |
589 mpcGtkPrintLabel(channelsLabel, "Channels: \%d", info.channels); | |
590 mpcGtkPrintLabel(lengthLabel, "Length: \%d:\%.2d", minutes, seconds); | |
591 mpcGtkPrintLabel(fileSizeLabel, "File size: \%d Bytes", info.total_file_length); | |
592 mpcGtkPrintLabel(trackPeakLabel, "Track Peak: \%5u", info.peak_title); | |
593 mpcGtkPrintLabel(trackGainLabel, "Track Gain: \%-+2.2f dB", 0.01 * info.gain_title); | |
594 mpcGtkPrintLabel(albumPeakLabel, "Album Peak: \%5u", info.peak_album); | |
595 mpcGtkPrintLabel(albumGainLabel, "Album Gain: \%-+5.2f dB", 0.01 * info.gain_album); | |
596 | |
597 MpcInfo tags = getTags(p_Filename); | |
598 gtk_entry_set_text(GTK_ENTRY(titleEntry), tags.title); | |
599 gtk_entry_set_text(GTK_ENTRY(artistEntry), tags.artist); | |
600 gtk_entry_set_text(GTK_ENTRY(albumEntry), tags.album); | |
601 gtk_entry_set_text(GTK_ENTRY(commentEntry), tags.comment); | |
602 gtk_entry_set_text(GTK_ENTRY(genreEntry), tags.genre); | |
603 char* entry = g_strdup_printf ("%d", tags.track); | |
604 gtk_entry_set_text(GTK_ENTRY(trackEntry), entry); | |
605 free(entry); | |
606 entry = g_strdup_printf ("%d", tags.year); | |
607 gtk_entry_set_text(GTK_ENTRY(yearEntry), entry); | |
608 free(entry); | |
609 entry = g_filename_display_name(p_Filename); | |
610 gtk_entry_set_text(GTK_ENTRY(fileEntry), entry); | |
611 free(entry); | |
612 freeTags(tags); | |
613 vfs_fclose(input); | |
614 } | |
615 else | |
616 { | |
617 char* temp = g_strdup_printf("[xmms-musepack] mpcFileInfoBox is unable to read tags from %s", p_Filename); | |
618 perror(temp); | |
619 free(temp); | |
620 } | |
621 | |
622 char* name = g_filename_display_basename(p_Filename); | |
623 char* text = g_strdup_printf("File Info - %s", name); | |
624 free(name); | |
625 gtk_window_set_title(GTK_WINDOW(infoBox), text); | |
626 free(text); | |
627 | |
628 gtk_widget_show_all(infoBox); | |
629 } | |
630 } | |
631 | |
632 static void mpcGtkPrintLabel(GtkWidget* widget, char* format,...) | |
633 { | |
634 va_list args; | |
635 | |
636 va_start(args, format); | |
637 char* temp = g_strdup_vprintf(format, args); | |
638 va_end(args); | |
639 | |
640 gtk_label_set_text(GTK_LABEL(widget), temp); | |
641 free(temp); | |
642 } | |
643 | |
644 static GtkWidget* mpcGtkTagLabel(char* p_Text, int a, int b, int c, int d, GtkWidget* p_Box) | |
645 { | |
646 GtkWidget* label = gtk_label_new(p_Text); | |
647 gtk_misc_set_alignment(GTK_MISC(label), 1, 0.5); | |
648 gtk_table_attach(GTK_TABLE(p_Box), label, a, b, c, d, GTK_FILL, GTK_FILL, 5, 5); | |
649 return label; | |
650 } | |
651 | |
652 static GtkWidget* mpcGtkTagEntry(int a, int b, int c, int d, int p_Size, GtkWidget* p_Box) | |
653 { | |
654 GtkWidget* entry; | |
655 if(p_Size == 0) | |
656 entry = gtk_entry_new(); | |
657 else | |
658 entry = gtk_entry_new_with_max_length(p_Size); | |
659 gtk_table_attach(GTK_TABLE(p_Box), entry, a, b, c, d, | |
660 (GtkAttachOptions) (GTK_FILL | GTK_EXPAND | GTK_SHRINK), | |
661 (GtkAttachOptions) (GTK_FILL | GTK_EXPAND | GTK_SHRINK), 0, 5); | |
662 return entry; | |
663 } | |
664 | |
665 static GtkWidget* mpcGtkLabel(GtkWidget* p_Box) | |
666 { | |
667 GtkWidget* label = gtk_label_new(""); | |
668 gtk_misc_set_alignment(GTK_MISC(label), 0, 0); | |
669 gtk_label_set_justify(GTK_LABEL(label), GTK_JUSTIFY_LEFT); | |
670 gtk_box_pack_start(GTK_BOX(p_Box), label, FALSE, FALSE, 0); | |
671 return label; | |
672 } | |
673 | |
674 static GtkWidget* mpcGtkButton(char* p_Text, GtkWidget* p_Box) | |
675 { | |
676 GtkWidget* button = gtk_button_new_with_label(p_Text); | |
677 GTK_WIDGET_SET_FLAGS(button, GTK_CAN_DEFAULT); | |
678 gtk_box_pack_start(GTK_BOX(p_Box), button, TRUE, TRUE, 0); | |
679 return button; | |
680 } | |
681 | |
682 static void removeTags(GtkWidget * w, gpointer data) | |
683 { | |
684 File oFile(gtk_entry_get_text(GTK_ENTRY(widgets.fileEntry))); | |
685 oFile.remove(); | |
686 oFile.save(); | |
687 closeInfoBox(NULL, NULL); | |
688 } | |
689 | |
690 static void saveTags(GtkWidget* w, gpointer data) | |
691 { | |
692 File oFile(gtk_entry_get_text(GTK_ENTRY(widgets.fileEntry))); | |
693 Tag* poTag = oFile.tag(); | |
694 | |
695 char* cAlbum = g_strdup(gtk_entry_get_text(GTK_ENTRY(widgets.albumEntry))); | |
696 char* cArtist = g_strdup(gtk_entry_get_text(GTK_ENTRY(widgets.artistEntry))); | |
697 char* cTitle = g_strdup(gtk_entry_get_text(GTK_ENTRY(widgets.titleEntry))); | |
698 char* cGenre = g_strdup(gtk_entry_get_text(GTK_ENTRY(widgets.genreEntry))); | |
699 char* cComment = g_strdup(gtk_entry_get_text(GTK_ENTRY(widgets.commentEntry))); | |
700 | |
701 const String album = String(cAlbum, TagLib::String::UTF8); | |
702 const String artist = String(cArtist, TagLib::String::UTF8); | |
703 const String title = String(cTitle, TagLib::String::UTF8); | |
704 const String genre = String(cGenre, TagLib::String::UTF8); | |
705 const String comment = String(cComment, TagLib::String::UTF8); | |
706 | |
707 poTag->setAlbum(album); | |
708 poTag->setArtist(artist); | |
709 poTag->setTitle(title); | |
710 poTag->setGenre(genre); | |
711 poTag->setComment(comment); | |
712 poTag->setYear(atoi(gtk_entry_get_text(GTK_ENTRY(widgets.yearEntry)))); | |
713 poTag->setTrack(atoi(gtk_entry_get_text(GTK_ENTRY(widgets.trackEntry)))); | |
714 | |
715 free(cAlbum); | |
716 free(cArtist); | |
717 free(cTitle); | |
718 free(cGenre); | |
719 free(cComment); | |
720 | |
721 oFile.save(); | |
722 closeInfoBox(NULL, NULL); | |
723 } | |
724 | |
725 static void closeInfoBox(GtkWidget* w, gpointer data) | |
726 { | |
727 gtk_widget_destroy(widgets.infoBox); | |
728 widgets.infoBox = NULL; | |
729 } | |
730 | |
731 static char* mpcGenerateTitle(const MpcInfo& p_Tags, char* p_Filename) | |
732 { | |
733 TitleInput* tuple = mpcGetSongTuple(p_Filename); | |
734 | |
735 char* title = xmms_get_titlestring (xmms_get_gentitle_format(), tuple); | |
736 if(!title) | |
737 title = g_strdup(tuple->file_name); | |
738 else if (!*title) | |
739 title = g_strdup(tuple->file_name); | |
740 | |
741 bmp_title_input_free(tuple); | |
742 return title; | |
743 } | |
744 | |
745 static void* endThread(char* p_FileName, VFSFile * p_FileHandle, bool release) | |
746 { | |
747 free(p_FileName); | |
748 if(release) | |
749 lockRelease(); | |
750 if(mpcDecoder.isError) | |
751 { | |
752 perror(mpcDecoder.isError); | |
753 free(mpcDecoder.isError); | |
754 mpcDecoder.isError = NULL; | |
755 } | |
756 setAlive(false); | |
757 if(p_FileHandle) | |
758 vfs_fclose(p_FileHandle); | |
759 if(track.display) | |
760 { | |
761 free(track.display); | |
762 track.display = NULL; | |
763 } | |
764 g_thread_exit(NULL); | |
765 return 0; | |
766 } | |
767 | |
768 static void* decodeStream(void* data) | |
769 { | |
770 lockAcquire(); | |
771 char* filename = static_cast<char*> (data); | |
772 VFSFile *input = vfs_fopen(filename, "rb"); | |
773 if (!input) | |
774 { | |
775 mpcDecoder.isError = g_strdup_printf("[xmms-musepack] decodeStream is unable to open %s", filename); | |
776 return endThread(filename, input, true); | |
777 } | |
778 | |
779 mpc_reader_file reader; | |
780 mpc_reader_setup_file_vfs(&reader, input); | |
781 | |
782 mpc_streaminfo info; | |
783 if (mpc_streaminfo_read(&info, &reader.reader) != ERROR_CODE_OK) | |
784 { | |
785 mpcDecoder.isError = g_strdup_printf("[xmms-musepack] decodeStream is unable to read %s", filename); | |
786 return endThread(filename, input, true); | |
787 } | |
788 | |
789 MpcInfo tags = getTags(filename); | |
790 track.display = mpcGenerateTitle(tags, filename); | |
791 track.length = static_cast<int> (1000 * mpc_streaminfo_get_length(&info)); | |
792 track.bitrate = static_cast<int> (info.average_bitrate); | |
793 track.sampleFreq = info.sample_freq; | |
794 track.channels = info.channels; | |
795 freeTags(tags); | |
796 | |
797 MpcPlugin.set_info(track.display, track.length, track.bitrate, track.sampleFreq, track.channels); | |
798 | |
799 mpc_decoder decoder; | |
800 mpc_decoder_setup(&decoder, &reader.reader); | |
801 if (!mpc_decoder_initialize(&decoder, &info)) | |
802 { | |
803 mpcDecoder.isError = g_strdup_printf("[xmms-musepack] decodeStream is unable to initialize decoder on %s", filename); | |
804 return endThread(filename, input, true); | |
805 } | |
806 | |
807 setReplaygain(info, decoder); | |
808 | |
809 MPC_SAMPLE_FORMAT sampleBuffer[MPC_DECODER_BUFFER_LENGTH]; | |
810 char xmmsBuffer[MPC_DECODER_BUFFER_LENGTH * 4]; | |
811 | |
812 if (!MpcPlugin.output->open_audio(FMT_S16_LE, track.sampleFreq, track.channels)) | |
813 { | |
814 mpcDecoder.isError = g_strdup_printf("[xmms-musepack] decodeStream is unable to open an audio output"); | |
815 return endThread(filename, input, true); | |
816 } | |
817 else | |
818 { | |
819 mpcDecoder.isOutput = true; | |
820 } | |
821 | |
822 lockRelease(); | |
823 | |
824 int counter = 2 * track.sampleFreq / 3; | |
825 while (isAlive()) | |
826 { | |
827 if (getOffset() != -1) | |
828 { | |
829 mpc_decoder_seek_seconds(&decoder, mpcDecoder.offset); | |
830 setOffset(-1); | |
831 } | |
832 | |
833 lockAcquire(); | |
834 short iPlaying = MpcPlugin.output->buffer_playing()? 1 : 0; | |
835 int iFree = MpcPlugin.output->buffer_free(); | |
836 if (!mpcDecoder.isPause && iFree >= ((1152 * 4) << iPlaying)) | |
837 { | |
838 unsigned status = processBuffer(sampleBuffer, xmmsBuffer, decoder); | |
839 if (status == (unsigned) (-1)) | |
840 { | |
841 mpcDecoder.isError = g_strdup_printf("[xmms-musepack] error from internal decoder on %s", filename); | |
842 return endThread(filename, input, true); | |
843 } | |
844 if (status == 0 && iPlaying == 0) | |
845 return endThread(filename, input, true); | |
846 | |
847 lockRelease(); | |
848 | |
849 if(pluginConfig.dynamicBitrate) | |
850 { | |
851 counter -= status; | |
852 if(counter < 0) | |
853 { | |
854 MpcPlugin.set_info(track.display, track.length, track.bitrate, track.sampleFreq, track.channels); | |
855 counter = 2 * track.sampleFreq / 3; | |
856 } | |
857 } | |
858 } | |
859 else | |
860 { | |
861 lockRelease(); | |
862 xmms_usleep(10000); | |
863 } | |
864 } | |
865 return endThread(filename, input, false); | |
866 } | |
867 | |
868 static int processBuffer(MPC_SAMPLE_FORMAT* sampleBuffer, char* xmmsBuffer, mpc_decoder& decoder) | |
869 { | |
870 mpc_uint32_t vbrAcc = 0; | |
871 mpc_uint32_t vbrUpd = 0; | |
872 | |
873 unsigned status = mpc_decoder_decode(&decoder, sampleBuffer, &vbrAcc, &vbrUpd); | |
874 copyBuffer(sampleBuffer, xmmsBuffer, status); | |
875 | |
876 if (pluginConfig.dynamicBitrate) | |
877 { | |
878 track.bitrate = static_cast<int> (vbrUpd * track.sampleFreq / 1152); | |
879 } | |
880 | |
881 produce_audio(MpcPlugin.output->written_time(), FMT_S16_LE, track.channels, status * 4, xmmsBuffer, NULL); | |
882 return status; | |
883 } | |
884 | |
885 static void setReplaygain(mpc_streaminfo& info, mpc_decoder& decoder) | |
886 { | |
887 if(!pluginConfig.replaygain && !pluginConfig.clipPrevention) | |
888 return; | |
889 | |
890 int peak = pluginConfig.albumGain ? info.peak_album : info.peak_title; | |
891 double gain = pluginConfig.albumGain ? info.gain_album : info.gain_title; | |
892 | |
893 if(!peak) | |
894 peak = 32767; | |
895 if(!gain) | |
896 gain = 1.; | |
897 | |
898 double clip = 32767. / peak; | |
899 gain = exp((M_LN10 / 2000.) * gain); | |
900 | |
901 if(pluginConfig.clipPrevention && !pluginConfig.replaygain) | |
902 gain = clip; | |
903 else if(pluginConfig.replaygain && pluginConfig.clipPrevention) | |
904 if(clip < gain) | |
905 gain = clip; | |
906 | |
907 mpc_decoder_scale_output(&decoder, gain); | |
908 } | |
909 | |
910 inline static void lockAcquire() | |
911 { | |
912 g_static_mutex_lock(&threadMutex); | |
913 } | |
914 | |
915 inline static void lockRelease() | |
916 { | |
917 g_static_mutex_unlock(&threadMutex); | |
918 } | |
919 | |
920 inline static bool isAlive() | |
921 { | |
922 lockAcquire(); | |
923 bool isAlive = mpcDecoder.isAlive; | |
924 lockRelease(); | |
925 return isAlive; | |
926 } | |
927 | |
928 inline static bool isPause() | |
929 { | |
930 lockAcquire(); | |
931 bool isPause = mpcDecoder.isPause; | |
932 lockRelease(); | |
933 return isPause; | |
934 } | |
935 | |
936 inline static void setAlive(bool isAlive) | |
937 { | |
938 lockAcquire(); | |
939 mpcDecoder.isAlive = isAlive; | |
940 lockRelease(); | |
941 } | |
942 | |
943 inline static double getOffset() | |
944 { | |
945 lockAcquire(); | |
946 double offset = mpcDecoder.offset; | |
947 lockRelease(); | |
948 return offset; | |
949 } | |
950 | |
951 inline static void setOffset(double offset) | |
952 { | |
953 lockAcquire(); | |
954 mpcDecoder.offset = offset; | |
955 lockRelease(); | |
956 } |