/* win32lib.c: bits and pieces for win32 and msvc.
Copyright 2006, 2011-2013 Akira Kakuto.
Copyright 1996, 1997, 1998, 1999 Fabrice Popineau.
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version.
This library 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
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License
along with this library; if not, see . */
#include
#include
#include
#include
FILE * win32_popen (const char *cmd, const char *fmode)
{
char mode[3];
mode[0] = fmode[0];
mode[1] = 'b';
mode[2] = '\0';
return _popen (cmd, mode);
}
int win32_pclose (FILE *f)
{
return _pclose (f);
}
/* large file support */
__int64
xftell64 (FILE *f, const char *filename)
{
__int64 where, filepos;
int fd;
fd = fileno(f);
if(f->_cnt < 0)
f->_cnt = 0;
if((filepos = _lseeki64(fd, (__int64)0, SEEK_CUR)) < (__int64)0) {
FATAL_PERROR(filename);
return (__int64)(-1);
}
if(filepos == (__int64)0)
where = (__int64)(f->_ptr - f->_base);
else
where = filepos - f->_cnt;
return where;
}
void
xfseek64 (FILE *f, __int64 offset, int wherefrom, const char *filename)
{
if(wherefrom == SEEK_CUR) {
offset += xftell64(f, filename);
wherefrom = SEEK_SET;
}
fflush(f);
if (_lseeki64(fileno(f), offset, wherefrom) < (__int64)0)
FATAL_PERROR(filename);
}
/* special TeXLive Ghostscript */
static int is_dir (char *buff)
{
struct stat stats;
return stat (buff, &stats) == 0 && S_ISDIR (stats.st_mode);
}
/*
TeXlive uses its own gs in
$SELFAUTOPARENT/tlpkg/tlgs
*/
void texlive_gs_init(void)
{
char *nptr, *path;
char tlgsbindir[512];
char tlgslibdir[512];
nptr = kpse_var_value("SELFAUTOPARENT");
if (nptr) {
strcpy(tlgsbindir, nptr);
strcat(tlgsbindir,"/tlpkg/tlgs");
if(is_dir(tlgsbindir)) {
strcpy(tlgslibdir, tlgsbindir);
strcat(tlgslibdir, "/lib;");
strcat(tlgslibdir, tlgsbindir);
strcat(tlgslibdir, "/fonts");
strcat(tlgsbindir, "/bin;");
free(nptr);
for(nptr = tlgsbindir; *nptr; nptr++) {
if(*nptr == '/') *nptr = '\\';
}
nptr = getenv("PATH");
path = (char *)malloc(strlen(nptr) + strlen(tlgsbindir) + 6);
strcpy(path, tlgsbindir);
strcat(path, nptr);
xputenv("PATH", path);
xputenv("GS_LIB", tlgslibdir);
}
}
}
/*
path name in char *input is changed into the long
name format and returned in char *buff.
return value: 0 if failed
1 if succeeded
*/
int getlongpath(char *buff, char *input, int len)
{
HANDLE hnd;
WIN32_FIND_DATA ffd;
int cnt = 0;
char *p, *q, *r;
buff[0] = '\0';
/*
temporarily change directory separators into back slashs
*/
for(p = input; *p; p++) {
if(*p == '/')
*p = '\\';
}
p = q = input;
r = buff;
/*
UNC name
*/
if(q[0] == '\\' && q[1] == '\\') {
cnt += 2;
if(cnt > len) return 0;
buff[0] = '/';
buff[1] = '/';
p += 2;
r += 2;
while(*p != '\\' && *p) {
if (IS_KANJI(p)) {
cnt++;
if(cnt > len) return 0;
*r++ = *p++;
}
cnt++;
if(cnt > len) return 0;
*r++ = *p++;
}
cnt++;
if(cnt > len) return 0;
*r++ = '/';
if(*p) p++;
while(*p != '\\' && *p) {
if (IS_KANJI(p)) {
cnt++;
if(cnt > len) return 0;
*r++ = *p++;
}
cnt++;
if(cnt > len) return 0;
*r++ = *p++;
}
cnt++;
if(cnt > len) return 0;
*r++ = '/';
*r= '\0';
if(*p) p++;
/*
drive name
*/
} else if(isalpha(q[0]) && q[1] == ':' && q[2] == '\\') {
*r++ = q[0];
*r++ = ':';
*r++ = '/';
*r = '\0';
p += 3;
cnt += 3;
if(cnt > len) return 0;
}
for( ; *p; p++) {
if(IS_KANJI(p)) {
p++;
continue;
}
if(*p == '\\') {
*p = '\0';
if((hnd = FindFirstFile(q, &ffd)) == INVALID_HANDLE_VALUE) {
return 0;
}
FindClose(hnd);
cnt += strlen(ffd.cFileName);
cnt++;
if(cnt > len) return 0;
strcat(buff, ffd.cFileName);
strcat(buff, "/");
*p = '\\';
}
}
/*
file itself
*/
if((hnd = FindFirstFile(q, &ffd)) == INVALID_HANDLE_VALUE) {
return 0;
}
FindClose(hnd);
cnt += strlen(ffd.cFileName);
if(cnt > len) return 0;
strcat(buff, ffd.cFileName);
return 1;
}
/* user info */
/* Adapted for XEmacs by David Hobley */
/* Sync'ed with Emacs 19.34.6 by Marc Paquette */
/* Adapted to fpTeX 0.4 by Fabrice Popineau */
char * get_home_directory()
{
char *p;
char *home = getenv("HOME");
if(!home)
home = getenv("USERPROFILE");
if(home) {
home = xstrdup(home);
for(p = home; *p; p++) {
if(IS_KANJI(p)) {
p++;
continue;
}
if(*p == '\\')
*p = '/';
}
}
return home;
}
int
kpathsea_getuid (kpathsea kpse)
{
return kpse->the_passwd.pw_uid;
}
int
kpathsea_getgid (kpathsea kpse)
{
return kpse->the_passwd.pw_gid;
}
struct passwd *
kpathsea_getpwuid (kpathsea kpse, int uid)
{
if (uid == kpse->the_passwd.pw_uid)
return &kpse->the_passwd;
return NULL;
}
struct passwd *
kpathsea_getpwnam (kpathsea kpse, char *name)
{
struct passwd *pw;
pw = kpathsea_getpwuid (kpse, kpathsea_getuid (kpse));
if (!pw)
return pw;
if (stricmp (name, pw->pw_name))
return NULL;
return pw;
}
void
kpathsea_init_user_info (kpathsea kpse)
{
char *home;
DWORD nSize = 256;
if (!GetUserName (kpse->the_passwd.pw_name, &nSize))
strcpy (kpse->the_passwd.pw_name, "unknown");
kpse->the_passwd.pw_uid = 123;
kpse->the_passwd.pw_gid = 123;
/* Ensure HOME and SHELL are defined. */
home = get_home_directory();
if (home) {
putenv(concat("HOME=", home));
}
else {
putenv ("HOME=c:/");
}
if (getenv ("SHELL") == NULL)
putenv ((GetVersion () & 0x80000000) ? "SHELL=command" : "SHELL=cmd");
{
/* If TEXMFTEMP is defined, then use it as the TEMP and TMP variables. */
char *p;
if ((p = getenv("TEXMFTEMP")) != NULL) {
putenv(concat("TEMP=", p));
putenv(concat("TMP=", p));
}
}
/* Set dir and shell from environment variables. */
strcpy (kpse->the_passwd.pw_dir, get_home_directory());
strcpy (kpse->the_passwd.pw_shell, getenv ("SHELL"));
}
/* win32_system */
static int is_include_space(const char *s)
{
char *p;
p = strchr(s, ' ');
if(p) return 1;
p = strchr(s, '\t');
if(p) return 1;
return 0;
}
int win32_system(const char *cmd)
{
const char *p;
char *q;
char *av[4];
int len, ret;
int spacep = 0;
if(cmd == NULL)
return 1;
av[0] = xstrdup("cmd.exe");
av[1] = xstrdup("/c");
len = strlen(cmd) + 3;
spacep = is_include_space(cmd);
av[2] = malloc(len);
q = av[2];
if(spacep)
*q++ = '"';
for(p = cmd; *p; p++, q++) {
if(*p == '\'')
*q = '"';
else
*q = *p;
}
if(spacep)
*q++ = '"';
*q = '\0';
av[3] = NULL;
ret = spawnvp(_P_WAIT, av[0], av);
free(av[0]);
free(av[1]);
free(av[2]);
return ret;
}
#if defined (KPSE_COMPAT_API)
int
getuid (void)
{
return kpathsea_getuid(kpse_def);
}
int
geteuid (void)
{
/* I could imagine arguing for checking to see whether the user is
in the Administrators group and returning a UID of 0 for that
case, but I don't know how wise that would be in the long run. */
return getuid ();
}
int
getgid (void)
{
return kpathsea_getgid (kpse_def);
}
int
getegid (void)
{
return getgid ();
}
struct passwd *
getpwuid (int uid)
{
return kpathsea_getpwuid (kpse_def, uid);
}
struct passwd *
getpwnam (char *name)
{
return kpathsea_getpwnam (kpse_def, name);
}
#endif /* KPSE_COMPAT_API */