mirror of
https://github.com/Pecusx/libretro-atari800.git
synced 2026-05-21 06:39:36 +02:00
initial commit
This commit is contained in:
@@ -0,0 +1,516 @@
|
||||
/*
|
||||
* util.c - utility functions
|
||||
*
|
||||
* Copyright (c) 2005 Piotr Fusik
|
||||
* Copyright (c) 2005-2013 Atari800 development team (see DOC/CREDITS)
|
||||
*
|
||||
* This file is part of the Atari800 emulator project which emulates
|
||||
* the Atari 400, 800, 800XL, 130XE, and 5200 8-bit computers.
|
||||
*
|
||||
* Atari800 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.
|
||||
*
|
||||
* Atari800 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 Atari800; if not, write to the Free Software
|
||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||
*/
|
||||
|
||||
#include "config.h"
|
||||
/* suppress -ansi -pedantic warning for fdopen: */
|
||||
#ifdef __STRICT_ANSI__
|
||||
#undef __STRICT_ANSI__
|
||||
#include <stdio.h>
|
||||
#define __STRICT_ANSI__ 1
|
||||
#else
|
||||
#include <stdio.h>
|
||||
#endif /* __STRICT_ANSI__ */
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <errno.h>
|
||||
#ifdef HAVE_SYS_STAT_H
|
||||
#include <sys/stat.h>
|
||||
#endif
|
||||
#ifdef HAVE_WINDOWS_H
|
||||
#include <windows.h>
|
||||
#endif
|
||||
#ifdef TIME_WITH_SYS_TIME
|
||||
# include <sys/time.h>
|
||||
# include <time.h>
|
||||
#else
|
||||
# ifdef HAVE_SYS_TIME_H
|
||||
# include <sys/time.h>
|
||||
# elif defined(HAVE_TIME_H)
|
||||
# include <time.h>
|
||||
# endif
|
||||
#endif
|
||||
|
||||
#include "atari.h"
|
||||
#include "platform.h"
|
||||
#include "util.h"
|
||||
|
||||
int Util_chrieq(char c1, char c2)
|
||||
{
|
||||
switch (c1 ^ c2) {
|
||||
case 0x00:
|
||||
return TRUE;
|
||||
case 0x20:
|
||||
return (c1 >= 'A' && c1 <= 'Z') || (c1 >= 'a' && c1 <= 'z');
|
||||
default:
|
||||
return FALSE;
|
||||
}
|
||||
}
|
||||
|
||||
#ifdef __STRICT_ANSI__
|
||||
int Util_stricmp(const char *str1, const char *str2)
|
||||
{
|
||||
int retval;
|
||||
|
||||
while((retval = tolower(*str1) - tolower(*str2++)) == 0)
|
||||
{
|
||||
if (*str1++ == '\0')
|
||||
break;
|
||||
}
|
||||
return retval;
|
||||
}
|
||||
#endif
|
||||
|
||||
char *Util_stpcpy(char *dest, const char *src)
|
||||
{
|
||||
size_t len = strlen(src);
|
||||
memcpy(dest, src, len + 1);
|
||||
return dest + len;
|
||||
}
|
||||
|
||||
#ifndef HAVE_STRNCPY
|
||||
char *Util_strncpy(char *dest, const char *src, size_t size) {
|
||||
while (size-- > 0) {
|
||||
if ((*dest++ = *src++) == '\0')
|
||||
break;
|
||||
}
|
||||
while (size-- > 0)
|
||||
*dest++ = '\0';
|
||||
return dest;
|
||||
}
|
||||
#endif
|
||||
|
||||
char *safe_strncpy(char *buffer, const char *source, int bufsize)
|
||||
{
|
||||
if (buffer == NULL) return NULL;
|
||||
if (bufsize > 0) {
|
||||
strncpy(buffer, source != NULL ? source : "", bufsize);
|
||||
buffer[bufsize-1] = '\0';
|
||||
}
|
||||
return buffer;
|
||||
}
|
||||
|
||||
char *Util_strlcpy(char *dest, const char *src, size_t size)
|
||||
{
|
||||
Util_strncpy(dest, src, size);
|
||||
dest[size - 1] = '\0';
|
||||
return dest;
|
||||
}
|
||||
|
||||
char *Util_strupper(char *s)
|
||||
{
|
||||
char *p;
|
||||
for (p = s; *p != '\0'; p++)
|
||||
if (*p >= 'a' && *p <= 'z')
|
||||
*p += 'A' - 'a';
|
||||
return s;
|
||||
}
|
||||
|
||||
char *Util_strlower(char *s)
|
||||
{
|
||||
char *p;
|
||||
for (p = s; *p != '\0'; p++)
|
||||
if (*p >= 'A' && *p <= 'Z')
|
||||
*p += 'a' - 'A';
|
||||
return s;
|
||||
}
|
||||
|
||||
void Util_chomp(char *s)
|
||||
{
|
||||
int len;
|
||||
|
||||
len = strlen(s);
|
||||
if (len >= 2 && s[len - 1] == '\n' && s[len - 2] == '\r')
|
||||
s[len - 2] = '\0';
|
||||
else if (len >= 1 && (s[len - 1] == '\n' || s[len - 1] == '\r'))
|
||||
s[len - 1] = '\0';
|
||||
}
|
||||
|
||||
void Util_trim(char *s)
|
||||
{
|
||||
char *p = s;
|
||||
char *q;
|
||||
/* skip leading whitespace */
|
||||
while (*p == ' ' || *p == '\t' || *p == '\r' || *p == '\n')
|
||||
p++;
|
||||
/* now p points at the first non-whitespace character */
|
||||
|
||||
if (*p == '\0') {
|
||||
/* only whitespace */
|
||||
*s = '\0';
|
||||
return;
|
||||
}
|
||||
|
||||
q = s + strlen(s);
|
||||
/* skip trailing whitespace */
|
||||
/* we have found p < q such that *p is non-whitespace,
|
||||
so this loop terminates with q >= p */
|
||||
do
|
||||
q--;
|
||||
while (*q == ' ' || *q == '\t' || *q == '\r' || *q == '\n');
|
||||
|
||||
/* now q points at the last non-whitespace character */
|
||||
/* cut off trailing whitespace */
|
||||
*++q = '\0';
|
||||
|
||||
/* move to string */
|
||||
memmove(s, p, q + 1 - p);
|
||||
}
|
||||
|
||||
int Util_sscandec(const char *s)
|
||||
{
|
||||
int result;
|
||||
if (*s == '\0')
|
||||
return -1;
|
||||
result = 0;
|
||||
for (;;) {
|
||||
if (*s >= '0' && *s <= '9')
|
||||
result = 10 * result + *s - '0';
|
||||
else if (*s == '\0')
|
||||
return result;
|
||||
else
|
||||
return -1;
|
||||
s++;
|
||||
}
|
||||
}
|
||||
|
||||
int Util_sscansdec(char const *s, int *dest)
|
||||
{
|
||||
int minus = FALSE;
|
||||
switch(*s) {
|
||||
case '-':
|
||||
minus = TRUE;
|
||||
/* Fallthrough! */
|
||||
case '+':
|
||||
++s;
|
||||
}
|
||||
*dest = Util_sscandec(s);
|
||||
if (*dest == -1)
|
||||
return FALSE;
|
||||
if (minus)
|
||||
*dest = -*dest;
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
int Util_sscandouble(char const *s, double *dest)
|
||||
{
|
||||
char *endptr;
|
||||
double result;
|
||||
|
||||
result = strtod(s, &endptr);
|
||||
if (endptr[0] != '\0' || errno == ERANGE)
|
||||
return FALSE;
|
||||
*dest = result;
|
||||
return TRUE;
|
||||
|
||||
}
|
||||
|
||||
int Util_sscanhex(const char *s)
|
||||
{
|
||||
int result;
|
||||
if (*s == '\0')
|
||||
return -1;
|
||||
result = 0;
|
||||
for (;;) {
|
||||
if (*s >= '0' && *s <= '9')
|
||||
result = 16 * result + *s - '0';
|
||||
else if (*s >= 'A' && *s <= 'F')
|
||||
result = 16 * result + *s - 'A' + 10;
|
||||
else if (*s >= 'a' && *s <= 'f')
|
||||
result = 16 * result + *s - 'a' + 10;
|
||||
else if (*s == '\0')
|
||||
return result;
|
||||
else
|
||||
return -1;
|
||||
s++;
|
||||
}
|
||||
}
|
||||
|
||||
int Util_sscanbool(const char *s)
|
||||
{
|
||||
if (*s == '0' && s[1] == '\0')
|
||||
return 0;
|
||||
if (*s == '1' && s[1] == '\0')
|
||||
return 1;
|
||||
return -1;
|
||||
}
|
||||
|
||||
#if !HAVE_ROUND
|
||||
double Util_round(double x)
|
||||
{
|
||||
return floor(x + 0.5);
|
||||
}
|
||||
#endif
|
||||
|
||||
void *Util_malloc(size_t size)
|
||||
{
|
||||
void *ptr = malloc(size);
|
||||
if (ptr == NULL) {
|
||||
Atari800_ErrExit();
|
||||
printf("Fatal error: out of memory\n");
|
||||
exit(1);
|
||||
}
|
||||
return ptr;
|
||||
}
|
||||
|
||||
void *Util_realloc(void *ptr, size_t size)
|
||||
{
|
||||
ptr = realloc(ptr, size);
|
||||
if (ptr == NULL) {
|
||||
Atari800_ErrExit();
|
||||
printf("Fatal error: out of memory\n");
|
||||
exit(1);
|
||||
}
|
||||
return ptr;
|
||||
}
|
||||
|
||||
char *Util_strdup(const char *s)
|
||||
{
|
||||
/* don't use strdup(): it is unavailable on WinCE */
|
||||
size_t size = strlen(s) + 1;
|
||||
char *ptr = (char *) Util_malloc(size);
|
||||
memcpy(ptr, s, size); /* faster than strcpy(ptr, s) */
|
||||
return ptr;
|
||||
}
|
||||
|
||||
void Util_splitpath(const char *path, char *dir_part, char *file_part)
|
||||
{
|
||||
const char *p;
|
||||
/* find the last Util_DIR_SEP_CHAR except the last character */
|
||||
for (p = path + strlen(path) - 2; p >= path; p--) {
|
||||
if (*p == Util_DIR_SEP_CHAR
|
||||
#ifdef DIR_SEP_BACKSLASH
|
||||
/* on DOSish systems slash can be also used as a directory separator */
|
||||
|| *p == '/'
|
||||
#endif
|
||||
) {
|
||||
if (dir_part != NULL) {
|
||||
int len = p - path;
|
||||
if (p == path || (p == path + 2 && path[1] == ':'))
|
||||
/* root dir: include Util_DIR_SEP_CHAR in dir_part */
|
||||
len++;
|
||||
memcpy(dir_part, path, len);
|
||||
dir_part[len] = '\0';
|
||||
}
|
||||
if (file_part != NULL)
|
||||
strcpy(file_part, p + 1);
|
||||
return;
|
||||
}
|
||||
}
|
||||
/* no Util_DIR_SEP_CHAR: current dir */
|
||||
if (dir_part != NULL)
|
||||
dir_part[0] = '\0';
|
||||
if (file_part != NULL)
|
||||
strcpy(file_part, path);
|
||||
}
|
||||
|
||||
void Util_catpath(char *result, const char *path1, const char *path2)
|
||||
{
|
||||
snprintf(result, FILENAME_MAX,
|
||||
path1[0] == '\0' || path2[0] == Util_DIR_SEP_CHAR || path1[strlen(path1) - 1] == Util_DIR_SEP_CHAR
|
||||
#ifdef DIR_SEP_BACKSLASH
|
||||
|| path2[0] == '/' || path1[strlen(path1) - 1] == '/'
|
||||
#endif
|
||||
? "%s%s" : "%s" Util_DIR_SEP_STR "%s", path1, path2);
|
||||
}
|
||||
|
||||
int Util_fileexists(const char *filename)
|
||||
{
|
||||
FILE *fp;
|
||||
fp = fopen(filename, "rb");
|
||||
if (fp == NULL)
|
||||
return FALSE;
|
||||
fclose(fp);
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
#ifdef HAVE_WINDOWS_H
|
||||
|
||||
int Util_direxists(const char *filename)
|
||||
{
|
||||
DWORD attr;
|
||||
#ifdef UNICODE
|
||||
WCHAR wfilename[FILENAME_MAX];
|
||||
if (MultiByteToWideChar(CP_ACP, 0, filename, -1, wfilename, FILENAME_MAX) <= 0)
|
||||
return FALSE;
|
||||
attr = GetFileAttributes(wfilename);
|
||||
#else
|
||||
attr = GetFileAttributes(filename);
|
||||
#endif /* UNICODE */
|
||||
if (attr == 0xffffffff)
|
||||
return FALSE;
|
||||
#ifdef _WIN32_WCE
|
||||
/* WinCE: Make sure user does not up-dir from the root */
|
||||
if (*filename == 0)
|
||||
return FALSE;
|
||||
#endif
|
||||
return (attr & FILE_ATTRIBUTE_DIRECTORY) ? TRUE : FALSE;
|
||||
}
|
||||
|
||||
#elif defined(HAVE_STAT)
|
||||
|
||||
int Util_direxists(const char *filename)
|
||||
{
|
||||
struct stat filestatus;
|
||||
return stat(filename, &filestatus) == 0 && (filestatus.st_mode & S_IFDIR);
|
||||
}
|
||||
|
||||
#else
|
||||
|
||||
int Util_direxists(const char *filename)
|
||||
{
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
#endif /* defined(HAVE_STAT) */
|
||||
|
||||
|
||||
int Util_flen(FILE *fp)
|
||||
{
|
||||
fseek(fp, 0, SEEK_END);
|
||||
return (int) ftell(fp);
|
||||
}
|
||||
|
||||
/* Creates a file that does not exist and fills in filename with its name.
|
||||
filename must point to FILENAME_MAX characters buffer which doesn't need
|
||||
to be initialized. */
|
||||
FILE *Util_uniqopen(char *filename, const char *mode)
|
||||
{
|
||||
/* We cannot simply call tmpfile(), because we don't want the file
|
||||
to be deleted when we close it, and we need the filename. */
|
||||
|
||||
#if defined(HAVE_MKSTEMP) && defined(HAVE_FDOPEN)
|
||||
/* this is the only implementation without a race condition */
|
||||
strcpy(filename, "a8XXXXXX");
|
||||
/* mkstemp() modifies the 'X'es and returns an open descriptor */
|
||||
return fdopen(mkstemp(filename), mode);
|
||||
#elif defined(HAVE_TMPNAM)
|
||||
/* tmpnam() is better than mktemp(), because it creates filenames
|
||||
in system's temporary directory. It is also more portable. */
|
||||
return fopen(tmpnam(filename), mode);
|
||||
#elif defined(HAVE_MKTEMP)
|
||||
strcpy(filename, "a8XXXXXX");
|
||||
/* mktemp() modifies the 'X'es and returns filename */
|
||||
return fopen(mktemp(filename), mode);
|
||||
#else
|
||||
/* Roll-your-own */
|
||||
int no;
|
||||
for (no = 0; no < 1000000; no++) {
|
||||
snprintf(filename, FILENAME_MAX, "a8%06d", no);
|
||||
if (!Util_fileexists(filename))
|
||||
return fopen(filename, mode);
|
||||
}
|
||||
return NULL;
|
||||
#endif
|
||||
}
|
||||
|
||||
#if defined(HAVE_WINDOWS_H) && defined(UNICODE)
|
||||
int Util_unlink(const char *filename)
|
||||
{
|
||||
WCHAR wfilename[FILENAME_MAX];
|
||||
#ifdef _WIN32_WCE
|
||||
char cwd[FILENAME_MAX];
|
||||
char fullfilename[FILENAME_MAX];
|
||||
if (filename[0] != '\\' && filename[0] != '/') {
|
||||
getcwd(cwd, FILENAME_MAX);
|
||||
Util_catpath(fullfilename, cwd, filename);
|
||||
if (MultiByteToWideChar(CP_ACP, 0, fullfilename, -1, wfilename, FILENAME_MAX) <= 0)
|
||||
return -1;
|
||||
}
|
||||
else
|
||||
#endif
|
||||
if (MultiByteToWideChar(CP_ACP, 0, filename, -1, wfilename, FILENAME_MAX) <= 0)
|
||||
return -1;
|
||||
return (DeleteFile(wfilename) != 0) ? 0 : -1;
|
||||
}
|
||||
#elif defined(HAVE_WINDOWS_H) && !defined(UNICODE)
|
||||
int Util_unlink(const char *filename)
|
||||
{
|
||||
return (DeleteFile(filename) != 0) ? 0 : -1;
|
||||
}
|
||||
#endif /* defined(HAVE_WINDOWS_H) && defined(UNICODE) */
|
||||
|
||||
double Util_time(void)
|
||||
{
|
||||
#ifdef SUPPORTS_PLATFORM_TIME
|
||||
return PLATFORM_Time();
|
||||
#elif defined(HAVE_WINDOWS_H)
|
||||
return GetTickCount() * 1e-3;
|
||||
#elif defined(DJGPP)
|
||||
/* DJGPP has gettimeofday, but it's not more accurate than uclock */
|
||||
return uclock() * (1.0 / UCLOCKS_PER_SEC);
|
||||
#elif defined(HAVE_GETTIMEOFDAY)
|
||||
struct timeval tp;
|
||||
gettimeofday(&tp, NULL);
|
||||
return tp.tv_sec + 1e-6 * tp.tv_usec;
|
||||
#elif defined(HAVE_UCLOCK)
|
||||
return uclock() * (1.0 / UCLOCKS_PER_SEC);
|
||||
#elif defined(HAVE_CLOCK)
|
||||
return clock() * (1.0 / CLK_TCK);
|
||||
#else
|
||||
#error No function found for Util_time()
|
||||
#endif
|
||||
}
|
||||
|
||||
/* FIXME: Ports should use SUPPORTS_PLATFORM_SLEEP and SUPPORTS_PLATFORM_TIME */
|
||||
/* and not this mess */
|
||||
|
||||
void Util_sleep(double s)
|
||||
{
|
||||
#ifdef SUPPORTS_PLATFORM_SLEEP
|
||||
PLATFORM_Sleep(s);
|
||||
#else /* !SUPPORTS_PLATFORM_SLEEP */
|
||||
if (s > 0) {
|
||||
#ifdef HAVE_WINDOWS_H
|
||||
Sleep((DWORD) (s * 1e3));
|
||||
#elif defined(DJGPP)
|
||||
/* DJGPP has usleep and select, but they don't work that good */
|
||||
/* XXX: find out why */
|
||||
double curtime = Util_time();
|
||||
while ((curtime + s) > Util_time());
|
||||
#elif defined(HAVE_NANOSLEEP)
|
||||
struct timespec ts;
|
||||
ts.tv_sec = 0;
|
||||
ts.tv_nsec = s * 1e9;
|
||||
nanosleep(&ts, NULL);
|
||||
#elif defined(HAVE_USLEEP)
|
||||
usleep(s * 1e6);
|
||||
#elif defined(__BEOS__)
|
||||
/* added by Walter Las for BeOS */
|
||||
snooze(s * 1e6);
|
||||
#elif defined(__EMX__)
|
||||
/* added by Brian Smith for os/2 */
|
||||
DosSleep(s);
|
||||
#elif defined(HAVE_SELECT)
|
||||
/* linux */
|
||||
struct timeval tp;
|
||||
tp.tv_sec = 0;
|
||||
tp.tv_usec = s * 1e6;
|
||||
select(1, NULL, NULL, NULL, &tp);
|
||||
#else
|
||||
double curtime = Util_time();
|
||||
while ((curtime + s) > Util_time());
|
||||
#endif
|
||||
}
|
||||
#endif /* !SUPPORTS_PLATFORM_SLEEP */
|
||||
}
|
||||
Reference in New Issue
Block a user