mirror of
https://github.com/ZDoom/acc.git
synced 2024-11-25 05:11:33 +00:00
408 lines
8.2 KiB
C
408 lines
8.2 KiB
C
|
|
//**************************************************************************
|
|
//**
|
|
//** misc.c
|
|
//**
|
|
//**************************************************************************
|
|
|
|
// HEADER FILES ------------------------------------------------------------
|
|
|
|
#ifdef __NeXT__
|
|
#include <libc.h>
|
|
#else
|
|
#include <fcntl.h>
|
|
#include <stdlib.h>
|
|
#if !defined(unix) && !defined(__APPLE__)
|
|
#include <io.h>
|
|
#endif
|
|
#endif
|
|
#ifdef __GNUC__
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
#endif
|
|
#ifdef _WIN32
|
|
#include <sys/stat.h>
|
|
#include <sys/types.h>
|
|
#endif
|
|
#include <stdio.h>
|
|
#include <stddef.h>
|
|
#include <stdarg.h>
|
|
#include <string.h>
|
|
#include <ctype.h>
|
|
#include "common.h"
|
|
#include "misc.h"
|
|
#include "error.h"
|
|
|
|
// MACROS ------------------------------------------------------------------
|
|
|
|
#ifndef O_BINARY
|
|
#define O_BINARY 0
|
|
#endif
|
|
|
|
// TYPES -------------------------------------------------------------------
|
|
|
|
// EXTERNAL FUNCTION PROTOTYPES --------------------------------------------
|
|
|
|
// PUBLIC FUNCTION PROTOTYPES ----------------------------------------------
|
|
|
|
// PRIVATE FUNCTION PROTOTYPES ---------------------------------------------
|
|
|
|
// EXTERNAL DATA DECLARATIONS ----------------------------------------------
|
|
|
|
extern boolean acs_BigEndianHost;
|
|
extern boolean acs_VerboseMode;
|
|
extern boolean acs_DebugMode;
|
|
extern FILE *acs_DebugFile;
|
|
|
|
// PUBLIC DATA DEFINITIONS -------------------------------------------------
|
|
|
|
// PRIVATE DATA DEFINITIONS ------------------------------------------------
|
|
|
|
// CODE --------------------------------------------------------------------
|
|
|
|
//==========================================================================
|
|
//
|
|
// MS_Alloc
|
|
//
|
|
//==========================================================================
|
|
|
|
void *MS_Alloc(size_t size, error_t error)
|
|
{
|
|
void *mem;
|
|
|
|
if((mem = malloc(size)) == NULL)
|
|
{
|
|
ERR_Exit(error, NO);
|
|
}
|
|
return mem;
|
|
}
|
|
|
|
//==========================================================================
|
|
//
|
|
// MS_Realloc
|
|
//
|
|
//==========================================================================
|
|
|
|
void *MS_Realloc(void *base, size_t size, error_t error)
|
|
{
|
|
void *mem;
|
|
|
|
if((mem = realloc(base, size)) == NULL)
|
|
{
|
|
ERR_Exit(error, NO);
|
|
}
|
|
return mem;
|
|
}
|
|
|
|
//==========================================================================
|
|
//
|
|
// MS_LittleUWORD
|
|
//
|
|
// Converts a host U_WORD (2 bytes) to little endian byte order.
|
|
//
|
|
//==========================================================================
|
|
|
|
U_WORD MS_LittleUWORD(U_WORD val)
|
|
{
|
|
if(acs_BigEndianHost == NO)
|
|
{
|
|
return val;
|
|
}
|
|
return ((val&255)<<8)+((val>>8)&255);
|
|
}
|
|
|
|
//==========================================================================
|
|
//
|
|
// MS_LittleUINT
|
|
//
|
|
// Converts a host U_INT (4 bytes) to little endian byte order.
|
|
//
|
|
//==========================================================================
|
|
|
|
U_INT MS_LittleUINT(U_INT val)
|
|
{
|
|
if(acs_BigEndianHost == NO)
|
|
{
|
|
return val;
|
|
}
|
|
return ((val&255)<<24)+(((val>>8)&255)<<16)+(((val>>16)&255)<<8)
|
|
+((val>>24)&255);
|
|
}
|
|
|
|
//==========================================================================
|
|
//
|
|
// MS_LoadFile
|
|
//
|
|
//==========================================================================
|
|
|
|
int MS_LoadFile(char *name, char **buffer)
|
|
{
|
|
int handle;
|
|
int size;
|
|
int count;
|
|
char *addr;
|
|
struct stat fileInfo;
|
|
|
|
if(strlen(name) >= MAX_FILE_NAME_LENGTH)
|
|
{
|
|
ERR_Exit(ERR_FILE_NAME_TOO_LONG, NO, name);
|
|
}
|
|
if((handle = open(name, O_RDONLY|O_BINARY, 0666)) == -1)
|
|
{
|
|
ERR_Exit(ERR_CANT_OPEN_FILE, NO, name);
|
|
}
|
|
if(fstat(handle, &fileInfo) == -1)
|
|
{
|
|
ERR_Exit(ERR_CANT_READ_FILE, NO, name);
|
|
}
|
|
size = fileInfo.st_size;
|
|
if((addr = malloc(size)) == NULL)
|
|
{
|
|
ERR_Exit(ERR_NONE, NO, "Couldn't malloc %d bytes for "
|
|
"file \"%s\".", size, name);
|
|
}
|
|
count = read(handle, addr, size);
|
|
close(handle);
|
|
if(count < size)
|
|
{
|
|
ERR_Exit(ERR_CANT_READ_FILE, NO, name);
|
|
}
|
|
*buffer = addr;
|
|
return size;
|
|
}
|
|
|
|
|
|
//==========================================================================
|
|
//
|
|
// MS_FileExists
|
|
//
|
|
// Pascal 21/11/08
|
|
//
|
|
//==========================================================================
|
|
boolean MS_FileExists(char *name)
|
|
{
|
|
struct stat info;
|
|
int ret = stat(name, &info);
|
|
return (ret == 0);
|
|
}
|
|
|
|
|
|
//==========================================================================
|
|
//
|
|
// MS_SaveFile
|
|
//
|
|
//==========================================================================
|
|
|
|
boolean MS_SaveFile(char *name, void *buffer, int length)
|
|
{
|
|
int handle;
|
|
int count;
|
|
|
|
handle = open(name, O_WRONLY|O_CREAT|O_TRUNC|O_BINARY, 0666);
|
|
if(handle == -1)
|
|
{
|
|
return FALSE;
|
|
}
|
|
count = write(handle, buffer, length);
|
|
close(handle);
|
|
if(count < length)
|
|
{
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
//==========================================================================
|
|
//
|
|
// MS_StrCmp
|
|
//
|
|
//==========================================================================
|
|
|
|
int MS_StrCmp(char *s1, char *s2)
|
|
{
|
|
for(; tolower(*s1) == tolower(*s2); s1++, s2++)
|
|
{
|
|
if(*s1 == '\0')
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
return tolower(*s1)-tolower(*s2);
|
|
}
|
|
|
|
//==========================================================================
|
|
//
|
|
// MS_StrLwr
|
|
//
|
|
//==========================================================================
|
|
|
|
char *MS_StrLwr(char *string)
|
|
{
|
|
char *c;
|
|
|
|
c = string;
|
|
while(*c)
|
|
{
|
|
*c = tolower(*c);
|
|
c++;
|
|
}
|
|
return string;
|
|
}
|
|
|
|
//==========================================================================
|
|
//
|
|
// MS_StrUpr
|
|
//
|
|
//==========================================================================
|
|
|
|
char *MS_StrUpr(char *string)
|
|
{
|
|
char *c;
|
|
|
|
c = string;
|
|
while(*c)
|
|
{
|
|
*c = toupper(*c);
|
|
c++;
|
|
}
|
|
return string;
|
|
}
|
|
|
|
//==========================================================================
|
|
//
|
|
// MS_SuggestFileExt
|
|
//
|
|
//==========================================================================
|
|
|
|
void MS_SuggestFileExt(char *base, char *extension)
|
|
{
|
|
char *search;
|
|
|
|
search = base+strlen(base)-1;
|
|
while(!MS_IsDirectoryDelimiter(*search) && search != base)
|
|
{
|
|
if(*search-- == '.')
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
strcat(base, extension);
|
|
}
|
|
|
|
//==========================================================================
|
|
//
|
|
// MS_IsDirectoryDelimiter
|
|
//
|
|
//==========================================================================
|
|
|
|
boolean MS_IsDirectoryDelimiter(char foo)
|
|
{
|
|
#if defined(_WIN32) || defined(__MSDOS__)
|
|
return foo == '/' || foo == '\\' || foo == ':';
|
|
#else
|
|
return foo == '/';
|
|
#endif
|
|
}
|
|
|
|
|
|
//==========================================================================
|
|
//
|
|
// MS_StripFileExt
|
|
//
|
|
//==========================================================================
|
|
|
|
void MS_StripFileExt(char *name)
|
|
{
|
|
char *search;
|
|
|
|
search = name+strlen(name)-1;
|
|
while(!MS_IsDirectoryDelimiter(*search) && search != name)
|
|
{
|
|
if(*search == '.')
|
|
{
|
|
*search = '\0';
|
|
return;
|
|
}
|
|
search--;
|
|
}
|
|
}
|
|
|
|
//==========================================================================
|
|
//
|
|
// MS_StripFilename
|
|
//
|
|
// [RH] This now leaves the directory delimiter in place.
|
|
//
|
|
//==========================================================================
|
|
|
|
boolean MS_StripFilename(char *name)
|
|
{
|
|
char *c;
|
|
|
|
c = name+strlen(name);
|
|
do
|
|
{
|
|
if(--c == name)
|
|
{ // No directory delimiter
|
|
return NO;
|
|
}
|
|
} while(!MS_IsDirectoryDelimiter(*c));
|
|
*(c+1) = 0;
|
|
return YES;
|
|
}
|
|
|
|
//==========================================================================
|
|
//
|
|
// MS_Message
|
|
//
|
|
//==========================================================================
|
|
|
|
void MS_Message(msg_t type, char *text, ...)
|
|
{
|
|
FILE *fp;
|
|
va_list argPtr;
|
|
|
|
if(type == MSG_VERBOSE && acs_VerboseMode == NO)
|
|
{
|
|
return;
|
|
}
|
|
fp = stdout;
|
|
if(type == MSG_DEBUG)
|
|
{
|
|
if(acs_DebugMode == NO)
|
|
{
|
|
return;
|
|
}
|
|
if(acs_DebugFile != NULL)
|
|
{
|
|
fp = acs_DebugFile;
|
|
}
|
|
}
|
|
if(text)
|
|
{
|
|
va_start(argPtr, text);
|
|
vfprintf(fp, text, argPtr);
|
|
va_end(argPtr);
|
|
}
|
|
}
|
|
|
|
//==========================================================================
|
|
//
|
|
// MS_IsPathAbsolute
|
|
//
|
|
// Pascal 30/11/08
|
|
//
|
|
//==========================================================================
|
|
|
|
boolean MS_IsPathAbsolute(char *name)
|
|
{
|
|
#if defined(_WIN32) || defined(__MSDOS__)
|
|
// In Windows, the second character must be : if it is an
|
|
// absolute path (the first character indicates the drive)
|
|
// or the first character is either / or \ for absolute path
|
|
if((name[0] != '\0') && (name[1] == ':'))
|
|
return TRUE;
|
|
#endif
|
|
// In Unix-land, the first character must be / for a root path
|
|
return MS_IsDirectoryDelimiter(name[0]);
|
|
}
|