Files
commandergenius/project/jni/application/atari800/src/util.c
T
2010-10-04 18:16:21 +03:00

420 lines
9.3 KiB
C

/*
* util.c - utility functions
*
* Copyright (c) 2005 Piotr Fusik
* Copyright (c) 2005 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>
#ifdef HAVE_SYS_STAT_H
#include <sys/stat.h>
#endif
#ifdef WIN32
#include <windows.h>
#endif
#include "atari.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__
/*
** STRICMP.C - Comapres strings, case-insensitive.
**
** public domain by Bob Stout
**
*/
/* from http://c.snippets.org/code/stricmp.c */
int Util_stricmp(const char *str1, const char *str2)
{
int retval = 0;
while (1)
{
retval = tolower(*str1++) - tolower(*str2++);
if (retval)
break;
if (*str1 && *str2)
continue;
else 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_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;
}
void *Util_malloc(size_t size)
{
void *ptr = malloc(size);
if (ptr == NULL) {
Atari800_Exit(FALSE);
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_Exit(FALSE);
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)
{
#ifdef HAVE_SNPRINTF
snprintf(result, FILENAME_MAX,
#else
sprintf(result,
#endif
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 WIN32
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++) {
sprintf(filename, "a8%06d", no);
if (!Util_fileexists(filename))
return fopen(filename, mode);
}
return NULL;
#endif
}
#if defined(WIN32) && 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(WIN32) && !defined(UNICODE)
int Util_unlink(const char *filename)
{
return (DeleteFile(filename) != 0) ? 0 : -1;
}
#endif /* defined(WIN32) && defined(UNICODE) */