Blame src/file/file_win32.c

Packit 5e46da
/*
Packit 5e46da
 * This file is part of libbluray
Packit 5e46da
 * Copyright (C) 2009-2010  Obliter0n
Packit 5e46da
 * Copyright (C) 2009-2010  John Stebbins
Packit 5e46da
 *
Packit 5e46da
 * This library is free software; you can redistribute it and/or
Packit 5e46da
 * modify it under the terms of the GNU Lesser General Public
Packit 5e46da
 * License as published by the Free Software Foundation; either
Packit 5e46da
 * version 2.1 of the License, or (at your option) any later version.
Packit 5e46da
 *
Packit 5e46da
 * This library is distributed in the hope that it will be useful,
Packit 5e46da
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 5e46da
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 5e46da
 * Lesser General Public License for more details.
Packit 5e46da
 *
Packit 5e46da
 * You should have received a copy of the GNU Lesser General Public
Packit 5e46da
 * License along with this library. If not, see
Packit 5e46da
 * <http://www.gnu.org/licenses/>.
Packit 5e46da
 */
Packit 5e46da
Packit 5e46da
#if HAVE_CONFIG_H
Packit 5e46da
#include "config.h"
Packit 5e46da
#endif
Packit 5e46da
Packit 5e46da
#if defined(__MINGW32__)
Packit 5e46da
/* ftello64() and fseeko64() prototypes from stdio.h */
Packit 5e46da
#   undef __STRICT_ANSI__
Packit 5e46da
#endif
Packit 5e46da
Packit 5e46da
#include "file.h"
Packit 5e46da
#include "util/macro.h"
Packit 5e46da
#include "util/logging.h"
Packit 5e46da
Packit 5e46da
#include <stdio.h>
Packit 5e46da
#include <stdlib.h>
Packit 5e46da
#include <inttypes.h>
Packit 5e46da
Packit 5e46da
#include <windows.h>
Packit 5e46da
Packit 5e46da
static void _file_close(BD_FILE_H *file)
Packit 5e46da
{
Packit 5e46da
    if (file) {
Packit 5e46da
        if (fclose((FILE *)file->internal)) {
Packit 5e46da
            BD_DEBUG(DBG_FILE | DBG_CRIT, "Error closing WIN32 file (%p)\n", (void*)file);
Packit 5e46da
        }
Packit 5e46da
Packit 5e46da
        BD_DEBUG(DBG_FILE, "Closed WIN32 file (%p)\n", (void*)file);
Packit 5e46da
Packit 5e46da
        X_FREE(file);
Packit 5e46da
    }
Packit 5e46da
}
Packit 5e46da
Packit 5e46da
static int64_t _file_seek(BD_FILE_H *file, int64_t offset, int32_t origin)
Packit 5e46da
{
Packit 5e46da
#if defined(__MINGW32__)
Packit 5e46da
    return fseeko64((FILE *)file->internal, offset, origin);
Packit 5e46da
#else
Packit 5e46da
    return _fseeki64((FILE *)file->internal, offset, origin);
Packit 5e46da
#endif
Packit 5e46da
}
Packit 5e46da
Packit 5e46da
static int64_t _file_tell(BD_FILE_H *file)
Packit 5e46da
{
Packit 5e46da
#if defined(__MINGW32__)
Packit 5e46da
    return ftello64((FILE *)file->internal);
Packit 5e46da
#else
Packit 5e46da
    return _ftelli64((FILE *)file->internal);
Packit 5e46da
#endif
Packit 5e46da
}
Packit 5e46da
Packit 5e46da
#if 0
Packit 5e46da
static int _file_eof(BD_FILE_H *file)
Packit 5e46da
{
Packit 5e46da
    return feof((FILE *)file->internal);
Packit 5e46da
}
Packit 5e46da
#endif
Packit 5e46da
Packit 5e46da
static int64_t _file_read(BD_FILE_H *file, uint8_t *buf, int64_t size)
Packit 5e46da
{
Packit 5e46da
    if (size > 0 && size < BD_MAX_SSIZE) {
Packit 5e46da
        return (int64_t)fread(buf, 1, (size_t)size, (FILE *)file->internal);
Packit 5e46da
    }
Packit 5e46da
Packit 5e46da
    BD_DEBUG(DBG_FILE | DBG_CRIT, "Ignoring invalid read of size %"PRId64" (%p)\n", size, (void*)file);
Packit 5e46da
    return 0;
Packit 5e46da
}
Packit 5e46da
Packit 5e46da
static int64_t _file_write(BD_FILE_H *file, const uint8_t *buf, int64_t size)
Packit 5e46da
{
Packit 5e46da
    if (size > 0 && size < BD_MAX_SSIZE) {
Packit 5e46da
        return (int64_t)fwrite(buf, 1, (size_t)size, (FILE *)file->internal);
Packit 5e46da
    }
Packit 5e46da
Packit 5e46da
    if (size == 0) {
Packit 5e46da
        if (fflush((FILE *)file->internal)) {
Packit 5e46da
            BD_DEBUG(DBG_FILE, "fflush() failed (%p)\n", (void*)file);
Packit 5e46da
            return -1;
Packit 5e46da
        }
Packit 5e46da
        return 0;
Packit 5e46da
    }
Packit 5e46da
Packit 5e46da
    BD_DEBUG(DBG_FILE | DBG_CRIT, "Ignoring invalid write of size %"PRId64" (%p)\n", size, (void*)file);
Packit 5e46da
    return 0;
Packit 5e46da
}
Packit 5e46da
Packit 5e46da
static BD_FILE_H *_file_open(const char* filename, const char *mode)
Packit 5e46da
{
Packit 5e46da
    BD_FILE_H *file;
Packit 5e46da
    FILE *fp;
Packit 5e46da
    wchar_t wfilename[MAX_PATH], wmode[8];
Packit 5e46da
Packit 5e46da
    if (!MultiByteToWideChar(CP_UTF8, MB_ERR_INVALID_CHARS, filename, -1, wfilename, MAX_PATH) ||
Packit 5e46da
        !MultiByteToWideChar(CP_UTF8, MB_ERR_INVALID_CHARS, mode, -1, wmode, 8)) {
Packit 5e46da
Packit 5e46da
        BD_DEBUG(DBG_FILE, "Error opening file %s\n", filename);
Packit 5e46da
        return NULL;
Packit 5e46da
    }
Packit 5e46da
Packit 5e46da
    fp = _wfopen(wfilename, wmode);
Packit 5e46da
    if (!fp) {
Packit 5e46da
        BD_DEBUG(DBG_FILE, "Error opening file %s\n", filename);
Packit 5e46da
        return NULL;
Packit 5e46da
    }
Packit 5e46da
Packit 5e46da
    file = calloc(1, sizeof(BD_FILE_H));
Packit 5e46da
    if (!file) {
Packit 5e46da
        BD_DEBUG(DBG_FILE | DBG_CRIT, "Error opening file %s (out of memory)\n", filename);
Packit 5e46da
        fclose(fp);
Packit 5e46da
        return NULL;
Packit 5e46da
    }
Packit 5e46da
Packit 5e46da
    file->internal = fp;
Packit 5e46da
    file->close    = _file_close;
Packit 5e46da
    file->seek     = _file_seek;
Packit 5e46da
    file->read     = _file_read;
Packit 5e46da
    file->write    = _file_write;
Packit 5e46da
    file->tell     = _file_tell;
Packit 5e46da
    //file->eof      = _file_eof;
Packit 5e46da
Packit 5e46da
    BD_DEBUG(DBG_FILE, "Opened WIN32 file %s (%p)\n", filename, (void*)file);
Packit 5e46da
    return file;
Packit 5e46da
}
Packit 5e46da
Packit 5e46da
BD_FILE_H* (*file_open)(const char* filename, const char *mode) = _file_open;
Packit 5e46da
Packit 5e46da
BD_FILE_OPEN file_open_default(void)
Packit 5e46da
{
Packit 5e46da
    return _file_open;
Packit 5e46da
}
Packit 5e46da
Packit 5e46da
int file_unlink(const char *file)
Packit 5e46da
{
Packit 5e46da
    wchar_t wfile[MAX_PATH];
Packit 5e46da
Packit 5e46da
    if (!MultiByteToWideChar(CP_UTF8, 0, file, -1, wfile, MAX_PATH)) {
Packit 5e46da
        return -1;
Packit 5e46da
    }
Packit 5e46da
Packit 5e46da
    return _wremove(wfile);
Packit 5e46da
}
Packit 5e46da
Packit 5e46da
int file_path_exists(const char *path)
Packit 5e46da
{
Packit 5e46da
    wchar_t wpath[MAX_PATH];
Packit 5e46da
Packit 5e46da
    if (!MultiByteToWideChar(CP_UTF8, 0, path, -1, wpath, MAX_PATH)) {
Packit 5e46da
        return -1;
Packit 5e46da
    }
Packit 5e46da
Packit 5e46da
    DWORD dwAttrib = GetFileAttributesW(wpath);
Packit 5e46da
    if (dwAttrib != INVALID_FILE_ATTRIBUTES) {
Packit 5e46da
        return 0;
Packit 5e46da
    }
Packit 5e46da
    return -1;
Packit 5e46da
}
Packit 5e46da
Packit 5e46da
int file_mkdir(const char *dir)
Packit 5e46da
{
Packit 5e46da
    wchar_t wdir[MAX_PATH];
Packit 5e46da
Packit 5e46da
    if (!MultiByteToWideChar(CP_UTF8, 0, dir, -1, wdir, MAX_PATH)) {
Packit 5e46da
        return -1;
Packit 5e46da
    }
Packit 5e46da
    if (!CreateDirectoryW(wdir, NULL))
Packit 5e46da
        return -1;
Packit 5e46da
    return 0;
Packit 5e46da
}