329 lines
6.4 KiB
C
329 lines
6.4 KiB
C
/* Copyright 1996-2002,2005,2007,2009,2011 Alain Knaff.
|
|
* This file is part of mtools.
|
|
*
|
|
* Mtools 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 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* Mtools 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 Mtools. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
* Miscellaneous routines.
|
|
*/
|
|
|
|
#include "sysincludes.h"
|
|
#include "msdos.h"
|
|
#include "stream.h"
|
|
#include "vfat.h"
|
|
#include "mtools.h"
|
|
|
|
|
|
void printOom(void)
|
|
{
|
|
fprintf(stderr, "Out of memory error");
|
|
}
|
|
|
|
char *get_homedir(void)
|
|
{
|
|
#ifndef OS_mingw32msvc
|
|
struct passwd *pw;
|
|
uid_t uid;
|
|
char *homedir;
|
|
char *username;
|
|
|
|
homedir = getenv ("HOME");
|
|
/*
|
|
* first we call getlogin.
|
|
* There might be several accounts sharing one uid
|
|
*/
|
|
if ( homedir )
|
|
return homedir;
|
|
|
|
pw = 0;
|
|
|
|
username = getenv("LOGNAME");
|
|
if ( !username )
|
|
username = getlogin();
|
|
if ( username )
|
|
pw = getpwnam( username);
|
|
|
|
if ( pw == 0 ){
|
|
/* if we can't getlogin, look up the pwent by uid */
|
|
uid = geteuid();
|
|
pw = getpwuid(uid);
|
|
}
|
|
|
|
/* we might still get no entry */
|
|
if ( pw )
|
|
return pw->pw_dir;
|
|
return 0;
|
|
#else
|
|
return getenv("HOME");
|
|
#endif
|
|
}
|
|
|
|
|
|
static void get_mcwd_file_name(char *file)
|
|
{
|
|
char *mcwd_path;
|
|
const char *homedir;
|
|
|
|
mcwd_path = getenv("MCWD");
|
|
if (mcwd_path == NULL || *mcwd_path == '\0'){
|
|
homedir= get_homedir();
|
|
if(!homedir)
|
|
homedir="/tmp";
|
|
strncpy(file, homedir, MAXPATHLEN-6);
|
|
file[MAXPATHLEN-6]='\0';
|
|
strcat( file, "/.mcwd");
|
|
} else {
|
|
strncpy(file, mcwd_path, MAXPATHLEN);
|
|
file[MAXPATHLEN]='\0';
|
|
}
|
|
}
|
|
|
|
void unlink_mcwd(void)
|
|
{
|
|
char file[MAXPATHLEN+1];
|
|
get_mcwd_file_name(file);
|
|
unlink(file);
|
|
}
|
|
|
|
FILE *open_mcwd(const char *mode)
|
|
{
|
|
struct MT_STAT sbuf;
|
|
char file[MAXPATHLEN+1];
|
|
time_t now;
|
|
|
|
get_mcwd_file_name(file);
|
|
if (*mode == 'r'){
|
|
if (MT_STAT(file, &sbuf) < 0)
|
|
return NULL;
|
|
/*
|
|
* Ignore the info, if the file is more than 6 hours old
|
|
*/
|
|
getTimeNow(&now);
|
|
if (now - sbuf.st_mtime > 6 * 60 * 60) {
|
|
fprintf(stderr,
|
|
"Warning: \"%s\" is out of date, removing it\n",
|
|
file);
|
|
unlink(file);
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
return fopen(file, mode);
|
|
}
|
|
|
|
|
|
|
|
void *safe_malloc(size_t size)
|
|
{
|
|
void *p;
|
|
|
|
p = malloc(size);
|
|
if(!p){
|
|
printOom();
|
|
exit(1);
|
|
}
|
|
return p;
|
|
}
|
|
|
|
void print_sector(const char *message, unsigned char *data, int size)
|
|
{
|
|
int col;
|
|
int row;
|
|
|
|
printf("%s:\n", message);
|
|
|
|
for(row = 0; row * 16 < size; row++){
|
|
printf("%03x ", row * 16);
|
|
for(col = 0; col < 16; col++)
|
|
printf("%02x ", data [row*16+col]);
|
|
for(col = 0; col < 16; col++) {
|
|
if(isprint(data [row*16+col]))
|
|
printf("%c", data [row*16+col]);
|
|
else
|
|
printf(".");
|
|
}
|
|
printf("\n");
|
|
}
|
|
}
|
|
|
|
#if (SIZEOF_TIME_T > SIZEOF_LONG) && defined (HAVE_STRTOLL)
|
|
# define STRTOTIME strtoll
|
|
#else
|
|
# define STRTOTIME strtol
|
|
#endif
|
|
|
|
time_t getTimeNow(time_t *now)
|
|
{
|
|
static int haveTime = 0;
|
|
static time_t sharedNow;
|
|
|
|
if(!haveTime) {
|
|
const char *source_date_epoch = getenv("SOURCE_DATE_EPOCH");
|
|
if (source_date_epoch) {
|
|
char *endptr;
|
|
time_t epoch =
|
|
STRTOTIME(source_date_epoch, &endptr, 10);
|
|
errno = 0;
|
|
|
|
if (endptr == source_date_epoch)
|
|
fprintf(stderr,
|
|
"SOURCE_DATE_EPOCH \"%s\" invalid\n",
|
|
source_date_epoch);
|
|
else if (errno != 0)
|
|
fprintf(stderr,
|
|
"SOURCE_DATE_EPOCH: strtoll: %s: %s\n",
|
|
strerror(errno), source_date_epoch);
|
|
else if (*endptr != '\0')
|
|
fprintf(stderr,
|
|
"SOURCE_DATE_EPOCH has trailing garbage \"%s\"\n",
|
|
endptr);
|
|
else {
|
|
sharedNow = epoch;
|
|
haveTime = 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
if(!haveTime) {
|
|
time(&sharedNow);
|
|
haveTime = 1;
|
|
}
|
|
if(now)
|
|
*now = sharedNow;
|
|
return sharedNow;
|
|
}
|
|
|
|
/* Convert a string to an offset. The string should be a number,
|
|
optionally followed by S (sectors), K (K-Bytes), M (Megabytes), G
|
|
(Gigabytes) */
|
|
off_t str_to_offset_with_end(const char *str, char **endp) {
|
|
char s;
|
|
off_t ofs;
|
|
|
|
*endp = NULL;
|
|
ofs = strtol(str, endp, 0);
|
|
s = **endp;
|
|
/* trailing char, see if it is a size specifier */
|
|
if (s == 's' || s == 'S') /* sector */
|
|
ofs <<= 9;
|
|
else if (s == 'k' || s == 'K') /* kb */
|
|
ofs <<= 10;
|
|
else if (s == 'm' || s == 'M') /* Mb */
|
|
ofs <<= 20;
|
|
else if (s == 'g' || s == 'G') /* Gb */
|
|
ofs <<= 30;
|
|
else
|
|
return ofs; /* invalid character */
|
|
(*endp)++;
|
|
return ofs;
|
|
}
|
|
|
|
/* Convert a string to a size. The string should be a number,
|
|
optionally followed by S (sectors), K (K-Bytes), M (Megabytes), G
|
|
(Gigabytes) */
|
|
mt_off_t str_to_off_with_end(const char *str, char **endp) {
|
|
char s;
|
|
mt_off_t siz;
|
|
|
|
*endp = NULL;
|
|
siz = strtol(str, endp, 0);
|
|
s = **endp;
|
|
/* trailing char, see if it is a size specifier */
|
|
if (s == 's' || s == 'S') /* sector */
|
|
siz <<= 9;
|
|
else if (s == 'k' || s == 'K') /* kb */
|
|
siz <<= 10;
|
|
else if (s == 'm' || s == 'M') /* Mb */
|
|
siz <<= 20;
|
|
else if (s == 'g' || s == 'G') /* Gb */
|
|
siz <<= 30;
|
|
else
|
|
return siz; /* invalid character */
|
|
(*endp)++;
|
|
return siz;
|
|
}
|
|
|
|
off_t str_to_offset(char *str) {
|
|
char *end;
|
|
off_t ofs = str_to_offset_with_end(str, &end);
|
|
if (ofs <= 0)
|
|
return 0; /* invalid or missing offset */
|
|
if (*end)
|
|
return 0; /* extra char, invalid */
|
|
return ofs;
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
#undef free
|
|
#undef malloc
|
|
|
|
static int total=0;
|
|
|
|
void myfree(void *ptr)
|
|
{
|
|
int *size = ((int *) ptr)-1;
|
|
total -= *size;
|
|
fprintf(stderr, "freeing %d bytes at %p total allocated=%d\n",
|
|
*size, ptr, total);
|
|
free(size);
|
|
}
|
|
|
|
void *mymalloc(size_t size)
|
|
{
|
|
int *ptr;
|
|
ptr = (int *)malloc(size+sizeof(int));
|
|
if(!ptr)
|
|
return 0;
|
|
*ptr = size;
|
|
ptr++;
|
|
total += size;
|
|
fprintf(stderr, "allocating %d bytes at %p total allocated=%d\n",
|
|
size, ptr, total);
|
|
return (void *) ptr;
|
|
}
|
|
|
|
void *mycalloc(size_t nmemb, size_t size)
|
|
{
|
|
void *ptr = mymalloc(nmemb * size);
|
|
if(!ptr)
|
|
return 0;
|
|
memset(ptr, 0, size);
|
|
return ptr;
|
|
}
|
|
|
|
void *myrealloc(void *ptr, size_t size)
|
|
{
|
|
int oldsize = ((int *)ptr) [-1];
|
|
void *new = mymalloc(size);
|
|
if(!new)
|
|
return 0;
|
|
memcpy(new, ptr, oldsize);
|
|
myfree(ptr);
|
|
return new;
|
|
}
|
|
|
|
char *mystrdup(char *src)
|
|
{
|
|
char *dest;
|
|
dest = mymalloc(strlen(src)+1);
|
|
if(!dest)
|
|
return 0;
|
|
strcpy(dest, src);
|
|
return dest;
|
|
}
|
|
|
|
#endif
|