Mercurial > audlegacy-plugins
view src/stdio/stdio.c @ 984:df7b09989aee trunk
[svn] - We got a new plugin, captain!
- FileWriter is the ultimate plugin for dumping audio to files. It
should be the successor of Disk Writer and Out-Lame, as it supports the
same output formats as those (WAVE and MP3). The main advantage of
having only one file dumping plugin for many formats is that not every
plugin has to think about file handling (where to write files to, how to
call them etc.) that much anymore.
- FileWriter is also very extensible - adding new output formats should
be very easy.
| author | mf0102 |
|---|---|
| date | Mon, 30 Apr 2007 14:16:32 -0700 |
| parents | f1642ee1115c |
| children | 183d03932c9e |
line wrap: on
line source
/* Audacious * Copyright (c) 2006 William Pitcock * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include <audacious/vfs.h> #include <audacious/plugin.h> #include <stdio.h> #include <unistd.h> #include <sys/stat.h> #include <sys/types.h> #include <string.h> static gchar * vfs_stdio_urldecode_path(const gchar * encoded_path) { const gchar *cur, *ext; gchar *path, *tmp; gint realchar; if (!encoded_path) return NULL; if (!str_has_prefix_nocase(encoded_path, "file:")) return NULL; cur = encoded_path + 5; if (str_has_prefix_nocase(cur, "//localhost")) cur += 11; if (*cur == '/') while (cur[1] == '/') cur++; tmp = g_malloc0(strlen(cur) + 1); while ((ext = strchr(cur, '%')) != NULL) { strncat(tmp, cur, ext - cur); ext++; cur = ext + 2; if (!sscanf(ext, "%2x", &realchar)) { /* Assume it is a literal '%'. Several file * managers send unencoded file: urls on drag * and drop. */ realchar = '%'; cur -= 2; } tmp[strlen(tmp)] = realchar; } path = g_strconcat(tmp, cur, NULL); g_free(tmp); return path; } VFSFile * stdio_vfs_fopen_impl(const gchar * path, const gchar * mode) { VFSFile *file; gchar *decpath; if (!path || !mode) return NULL; decpath = vfs_stdio_urldecode_path(path); file = g_new(VFSFile, 1); file->handle = fopen(decpath != NULL ? decpath : path, mode); if (decpath != NULL) g_free(decpath); if (file->handle == NULL) { g_free(file); file = NULL; } return file; } gint stdio_vfs_fclose_impl(VFSFile * file) { gint ret = 0; if (file == NULL) return -1; if (file->handle) { FILE *handle = (FILE *) file->handle; if (fclose(handle) != 0) ret = -1; } return ret; } size_t stdio_vfs_fread_impl(gpointer ptr, size_t size, size_t nmemb, VFSFile * file) { FILE *handle; if (file == NULL) return 0; handle = (FILE *) file->handle; return fread(ptr, size, nmemb, handle); } size_t stdio_vfs_fwrite_impl(gconstpointer ptr, size_t size, size_t nmemb, VFSFile * file) { FILE *handle; if (file == NULL) return 0; handle = (FILE *) file->handle; return fwrite(ptr, size, nmemb, handle); } gint stdio_vfs_getc_impl(VFSFile *stream) { FILE *handle = (FILE *) stream->handle; return getc( handle ); } gint stdio_vfs_ungetc_impl(gint c, VFSFile *stream) { FILE *handle = (FILE *) stream->handle; return ungetc( c , handle ); } gint stdio_vfs_fseek_impl(VFSFile * file, glong offset, gint whence) { FILE *handle; if (file == NULL) return 0; handle = (FILE *) file->handle; return fseek(handle, offset, whence); } void stdio_vfs_rewind_impl(VFSFile * file) { FILE *handle; if (file == NULL) return; handle = (FILE *) file->handle; rewind(handle); } glong stdio_vfs_ftell_impl(VFSFile * file) { FILE *handle; if (file == NULL) return 0; handle = (FILE *) file->handle; return ftell(handle); } gboolean stdio_vfs_feof_impl(VFSFile * file) { FILE *handle; if (file == NULL) return FALSE; handle = (FILE *) file->handle; return (gboolean) feof(handle); } gint stdio_vfs_truncate_impl(VFSFile * file, glong size) { FILE *handle; if (file == NULL) return -1; handle = (FILE *) file->handle; return ftruncate(fileno(handle), size); } off_t stdio_vfs_fsize_impl(VFSFile * file) { FILE *handle; struct stat s; if (file == NULL) return -1; handle = (FILE *) file->handle; if (-1 == fstat(fileno(handle), &s)) return -1; return s.st_size; } VFSConstructor file_const = { "file://", stdio_vfs_fopen_impl, stdio_vfs_fclose_impl, stdio_vfs_fread_impl, stdio_vfs_fwrite_impl, stdio_vfs_getc_impl, stdio_vfs_ungetc_impl, stdio_vfs_fseek_impl, stdio_vfs_rewind_impl, stdio_vfs_ftell_impl, stdio_vfs_feof_impl, stdio_vfs_truncate_impl, stdio_vfs_fsize_impl }; VFSConstructor default_const = { "/", stdio_vfs_fopen_impl, stdio_vfs_fclose_impl, stdio_vfs_fread_impl, stdio_vfs_fwrite_impl, stdio_vfs_getc_impl, stdio_vfs_ungetc_impl, stdio_vfs_fseek_impl, stdio_vfs_rewind_impl, stdio_vfs_ftell_impl, stdio_vfs_feof_impl, stdio_vfs_truncate_impl, stdio_vfs_fsize_impl }; static void init(void) { vfs_register_transport(&default_const); vfs_register_transport(&file_const); } static void cleanup(void) { #if 0 vfs_unregister_transport(&default_const); vfs_unregister_transport(&file_const); #endif } LowlevelPlugin llp_stdio = { NULL, NULL, "file:// URI Transport", init, cleanup, }; LowlevelPlugin *get_lplugin_info(void) { return &llp_stdio; }
