summaryrefslogtreecommitdiff
path: root/plugins/gme
diff options
context:
space:
mode:
authorGravatar Alexey Yakovenko <waker@users.sourceforge.net>2014-02-14 20:16:29 +0100
committerGravatar Alexey Yakovenko <waker@users.sourceforge.net>2014-02-14 20:16:29 +0100
commit8cda195231d378f9fb918c4da0b3ada3078c1c75 (patch)
tree2cd5f96bcc858cbf23fc2f721f6f91d0013b6c45 /plugins/gme
parent15de2eca7b2689070e80ff980bc766cab9c2ac55 (diff)
gme: adding gzipped modules directly from vfs folders (bug #965)
Diffstat (limited to 'plugins/gme')
-rw-r--r--plugins/gme/cgme.c49
1 files changed, 43 insertions, 6 deletions
diff --git a/plugins/gme/cgme.c b/plugins/gme/cgme.c
index 53fa6eb8..557ec13f 100644
--- a/plugins/gme/cgme.c
+++ b/plugins/gme/cgme.c
@@ -21,6 +21,9 @@
3. This notice may not be removed or altered from any source distribution.
*/
+#ifdef HAVE_CONFIG_H
+# include "../../config.h"
+#endif
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
@@ -28,10 +31,19 @@
#include "gme/gme.h"
#include <zlib.h>
#include "../../deadbeef.h"
+#if HAVE_SYS_CDEFS_H
+#include <sys/cdefs.h>
+#endif
+#if HAVE_SYS_SYSLIMITS_H
+#include <sys/syslimits.h>
+#endif
//#define trace(...) { fprintf(stderr, __VA_ARGS__); }
#define trace(fmt,...)
+// how big vgz can be?
+#define MAX_REMOTE_GZ_FILE 0x100000
+
static DB_decoder_t plugin;
static DB_functions_t *deadbeef;
static int conf_fadeout = 10;
@@ -55,14 +67,38 @@ cgme_open (uint32_t hint) {
static int
read_gzfile (const char *fname, char **buffer, int *size) {
- FILE *fp = fopen (fname, "rb");
+ int fd = -1;
+ DB_FILE *fp = deadbeef->fopen (fname);
if (!fp) {
- trace ("failed to fopen %s\n", fname);
+ trace ("gme read_gzfile: failed to fopen %s\n", fname);
return -1;
}
- fseek (fp, 0, SEEK_END);
- size_t sz = ftell (fp);
- fclose (fp);
+
+ int64_t sz = deadbeef->fgetlength (fp);
+ if (fp->vfs && fp->vfs->plugin.id && strcmp (fp->vfs->plugin.id, "vfs_stdio") && sz > 0 && sz <= MAX_REMOTE_GZ_FILE) {
+ trace ("gme read_gzfile: reading %s of size %lld and writing to temp file\n", fname, sz);
+ char buffer[sz];
+ if (sz == deadbeef->fread (buffer, 1, sz, fp)) {
+ const char *tmp = getenv ("TMPDIR");
+ if (!tmp) {
+ tmp = "/tmp";
+ }
+ char nm[PATH_MAX];
+ snprintf (nm, sizeof (nm), "%s/ddbgmeXXXXXX.vgz", tmp);
+ fd = mkstemps (nm, 4);
+ if (fd == -1 || sz != write (fd, buffer, sz)) {
+ trace ("gme read_gzfile: failed to write temp file\n");
+ if (fd != -1) {
+ close (fd);
+ }
+ }
+ if (fd != -1) {
+ lseek (fd, 0, SEEK_SET);
+ }
+ }
+ }
+
+ deadbeef->fclose (fp);
sz *= 2;
int readsize = sz;
@@ -71,7 +107,7 @@ read_gzfile (const char *fname, char **buffer, int *size) {
return -1;
}
- gzFile gz = gzopen (fname, "rb");
+ gzFile gz = fd == -1 ? gzopen (fname, "rb") : gzdopen (fd, "r");
if (!gz) {
trace ("failed to gzopen %s\n", fname);
return -1;
@@ -84,6 +120,7 @@ read_gzfile (const char *fname, char **buffer, int *size) {
if (nb < 0) {
free (*buffer);
trace ("failed to gzread from %s\n", fname);
+ gzclose (gz);
return -1;
}
if (nb > 0) {