beetle-psx-libretro/mednafen/file.c
2015-07-25 18:06:37 +02:00

177 lines
3.5 KiB
C

/* Mednafen - Multi-system Emulator
*
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
#include <stdio.h>
#include <stdint.h>
#include <stdlib.h>
#include <stdarg.h>
#include <string.h>
#include "file.h"
#include "mednafen-endian.h"
struct MDFNFILE *file_open(const char *path)
{
const char *ld;
FILE *fp;
struct MDFNFILE *file = (struct MDFNFILE*)calloc(1, sizeof(*file));
if (!file)
return NULL;
fp = fopen(path, "rb");
if (!fp)
goto error;
fseek(fp, 0, SEEK_SET);
fseek((FILE *)fp, 0, SEEK_END);
file->size = ftell((FILE *)fp);
fseek((FILE *)fp, 0, SEEK_SET);
if (!(file->data = (uint8_t*)malloc(file->size)))
goto error;
fread(file->data, 1, file->size, (FILE *)fp);
ld = (const char*)strrchr(path, '.');
file->ext = strdup(ld ? ld + 1 : "");
return file;
error:
if (fp)
fclose((FILE*)fp);
if (file)
free(file);
return NULL;
}
int file_close(struct MDFNFILE *file)
{
if (!file)
return 0;
if (file->ext)
free(file->ext);
file->ext = NULL;
if (file->data)
free(file->data);
file->data = NULL;
free(file);
return 1;
}
uint64_t file_read(struct MDFNFILE *file, void *ptr,
size_t element_size, size_t nmemb)
{
uint32_t total = element_size * nmemb;
if (file->location >= file->size)
return 0;
if ((file->location + total) > file->size)
{
int64_t ak = file->size - file->location;
memcpy((uint8_t*)ptr, file->data + file->location, ak);
file->location = file->size;
return(ak / element_size);
}
memcpy((uint8_t*)ptr, file->data + file->location, total);
file->location += total;
return nmemb;
}
int file_seek(struct MDFNFILE *file, int64_t offset, int whence)
{
switch(whence)
{
case SEEK_SET:
if (offset >= file->size)
return -1;
file->location = offset;
break;
case SEEK_CUR:
if ((offset + file->location) > file->size)
return -1;
file->location += offset;
break;
}
return 0;
}
int file_read16le(struct MDFNFILE *file, uint16_t *val)
{
if ((file->location + 2) > file->size)
return 0;
*val = MDFN_de16lsb(file->data + file->location);
file->location += 2;
return 1;
}
int file_read32le(struct MDFNFILE *file, uint32_t *val)
{
if ((file->location + 4) > file->size)
return 0;
*val = MDFN_de32lsb(file->data + file->location);
file->location += 4;
return 1;
}
char *file_fgets(struct MDFNFILE *file, char *s, int len)
{
int pos = 0;
if (!len)
return(NULL);
if (file->location >= len)
return(NULL);
while(pos < (len - 1) && file->location < len)
{
int v = file->data[file->location];
s[pos] = v;
file->location++;
pos++;
if (v == '\n')
break;
}
if (len)
s[pos] = 0;
return s;
}