/* -*-c-*- -------------- mix_file.c : * Implementation of the functions declared in mix_file.h * ------------------------------------------------------------------ * Copyright (C) 2000, 2001, 2007 Free Software Foundation, Inc. * * 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 3 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 <string.h> #include "xmix_io.h" #include "mix_file.h" const gchar *MIX_SRC_DEFEXT = ".mixal", *MIX_LIST_DEFEXT = ".mls", *MIX_CODE_DEFEXT = ".mix"; /* file names completions */ #define needs_completion_(name,defext) \ ( strcmp(name + strlen(name) - strlen(defext), defext) != 0 ) #define add_completion_(name, defext) \ g_strconcat(name, defext, NULL) /* The actual definition of mix_file_t */ struct mix_file_t { mix_iochannel_t parent; gchar *base_name; gchar *ext; }; /* Creation/destruction of files */ static mix_file_t * open_file_(const gchar *name, mix_fmode_t mode) { mix_file_t *result; FILE *file; const gchar *fmode = fmode_to_type_ (mode); /* if the read/write file already exists, open in r+ mode */ if (mode == mix_io_RDWRT && (file = fopen (name, "r"))) { fmode = "r+"; fclose (file); } result = g_new(mix_file_t, 1); file = fopen(name, fmode); if ( file == NULL ) { g_free (result); return NULL; } io_init_from_file_(MIX_IOCHANNEL(result), file); return result; } mix_file_t * mix_file_new(const gchar *name, mix_fmode_t mode) { mix_file_t *result; gchar *bname; if ( name == NULL ) return NULL; bname = g_strdup(name); if ( bname == NULL ) return NULL; result = open_file_(name, mode); if ( result == NULL ) { g_free(bname); return NULL; } result->base_name = bname; result->ext = NULL; return result; } /* creates a file adding to its name the defext if missing */ mix_file_t * mix_file_new_with_def_ext(const gchar *name, mix_fmode_t mode, const gchar *defext) { const gchar *real_name; mix_file_t *result; if ( name == NULL ) return NULL; if ( defext == NULL ) return mix_file_new(name, mode); real_name = needs_completion_(name, defext) ? add_completion_(name, defext) : name; result = open_file_(real_name, mode); if ( real_name != name ) g_free((void *)real_name); if ( result == NULL ) return NULL; result->ext = g_strdup(defext); if ( needs_completion_(name, defext) ) result->base_name = g_strdup(name); else result->base_name = g_strndup(name, strlen(name) - strlen(defext)); if ( result->ext == NULL || result->base_name == NULL ) { mix_file_delete(result); return NULL; } return result; } void mix_file_delete(mix_file_t *file) { g_return_if_fail(file != NULL); io_close_(MIX_IOCHANNEL(file)); if (file->base_name) g_free(file->base_name); if (file->ext) g_free(file->ext); g_free(file); } /* convert to a standard FILE */ extern FILE * mix_file_to_FILE(const mix_file_t *file) { if ( file == NULL ) return NULL; return io_get_FILE_(file); } /* complete a name with an extension, if needed */ gchar * mix_file_complete_name (const gchar *name, const gchar *extension) { if (!name) return NULL; if (!extension || !needs_completion_ (name, extension)) return g_strdup (name); return add_completion_ (name, extension); } /* Get the base name and extension of file */ const gchar * mix_file_base_name(const mix_file_t *file) { g_return_val_if_fail(file != NULL, NULL); return file->base_name; } const gchar * mix_file_extension(const mix_file_t *file) { g_return_val_if_fail(file != NULL, NULL); return file->ext; }