#include <stdlib.h>
#include <string.h>
#include <sys/stat.h>
#include <dirent.h>
#include <stdio.h>
#include <io.h>
#include <fcntl.h>
#include <process.h>
static char *concat();
static char *concat3();
static char *concat4();
static int onlyonedir;
static int atleastone;
static char *fixeddirs, *origdirs;
/* Convert all /'s to \'s */
char *
slash2slash (dirname)
char *dirname;
{
int i;
for (i=0; dirname[i]; i++)
if (dirname [i] == '/')
dirname [i] = '\\';
return dirname;
}
/* Examine each directory component of a path and create the directory */
int
mkdirpath (dirpath)
char *dirpath;
{
char *ndirpath = strdup (dirpath);
char *bp, *fp;
fp = bp = ndirpath;
while (bp)
{
bp = strchr (fp, '\\');
if (bp)
{
*bp = 0;
_mkdir (ndirpath);
*bp = '\\';
fp = ++bp;
}
else
_mkdir (ndirpath);
}
}
/* Construct a relative directory path from a given path by removing the
leading slash, if it exists and changing a drive letter from X: to X-. */
char *
newname (olddirname)
char *olddirname;
{
char *newname = strdup (olddirname);
if ((strlen (newname) >= 2)
&& (isalpha (newname[0]) && newname[1] == ':'))
newname [1] = '-';
else if ((strlen (newname) >= 1)
&& (newname [0] == '/' || newname [0] == '\\'))
newname = &newname[1];
return newname;
}
/* Run the sed script on one header file. If no modifications were made, then
delete the newly created file. */
int
doheader (oneheader, outheader, oldsize)
char *oneheader, *outheader;
int oldsize;
{
char *newbuff, *oldbuff;
char *newheader = concat3 ("include", "\\", newname (outheader));
struct _stat newstatbuf;
int newdesc, olddesc;
int i;
system (concat4 ("sed -f fixinc-nt.sed ", oneheader, " > ", newheader));
_stat (newheader, &newstatbuf);
if (oldsize != newstatbuf.st_size)
{
atleastone = 1;
printf ("Fixing: %s\n", oneheader);
return 0;
}
oldbuff = malloc (oldsize);
newbuff = malloc (newstatbuf.st_size);
olddesc = open (oneheader, _O_RDONLY | _O_BINARY);
newdesc = open (newheader, _O_RDONLY | _O_BINARY);
read (olddesc, oldbuff, oldsize);
read (newdesc, newbuff, newstatbuf.st_size);
close (olddesc);
close (newdesc);
for (i=0; i<oldsize; i++)
{
if (oldbuff [i] != newbuff [i])
{
free (oldbuff);
free (newbuff);
atleastone = 1;
printf ("Fixing: %s\n", oneheader);
return 0;
}
}
free (oldbuff);
free (newbuff);
unlink (newheader);
return 0;
}
/* Examine the contents of a directory and call doheader () for a regular file
and recursively call dodir () for an enclosed directory. */
int
dodir (indir, outdir)
char *indir, *outdir;
{
DIR *dir;
struct dirent *dire;
struct _stat statbuf;
char *intempbuf, *outtempbuf;
dir = opendir (indir);
if (!dir) return 0;
mkdirpath (concat3 ("include", "\\", newname (outdir)));
while ((dire = readdir (dir)))
{
if (dire->d_name[0] == '.')
continue;
intempbuf = slash2slash (concat3 (indir, "\\", dire->d_name));
outtempbuf = slash2slash (concat3 (outdir, "\\", dire->d_name));
_stat (intempbuf, &statbuf);
/* If directory ... */
if (statbuf.st_mode & _S_IFDIR)
dodir (intempbuf, outtempbuf);
/* If regular file ... */
if (statbuf.st_mode & _S_IFREG)
doheader (intempbuf, outtempbuf, statbuf.st_size);
}
closedir (dir);
return 0;
}
/* Retrieve the value of the Include environment variable, copy it into a
temporary and append a semi-colon for book-keeping purposes. Then call
dodir () for each complete directory that is named therein. If there is
only one directory, then direct the output to use include\. as the
root instead of include/<directory path>, where <directory path> is a path
constructed from the path named in the Include environment variable.
I.e. if Include=C:\MSTOOLS\Include;D:\MSVC20\Include then the modified
header files will be in include\C-\MSTOOLS\Include and
include\D-\MSVC20\Include. However if Include=C:\MSTOOLS\Include then the
modified files will be in include\. */
int
main ()
{
char *fp, *bp, *foobar;
char *incvar = getenv ("Include");
int varlen = 0;
struct _stat statbuf;
if (incvar == NULL) return 0;
varlen = strlen (incvar);
foobar = (char *) malloc (varlen + 2);
strcpy (foobar, incvar);
foobar = slash2slash (foobar);
if (foobar [varlen-1] != ';') strcat (foobar, ";");
fp = bp = foobar;
if (strchr (fp, ';') == strrchr (fp, ';'))
onlyonedir = 1;
else
onlyonedir = 0;
fixeddirs = strdup(".\\include");
origdirs = strdup("");
while (bp)
{
bp = strchr (fp, ';');
if (bp)
{
*bp = 0;
_stat (fp, &statbuf);
if (statbuf.st_mode & _S_IFDIR)
{
atleastone = 0;
if (onlyonedir)
dodir (fp, ".");
else
dodir (fp, fp);
if (atleastone && !onlyonedir)
{
origdirs = concat3 (origdirs, ";", fp);
fixeddirs = concat3 (fixeddirs, ";",
concat3 (".\\include", "\\", newname(fp)));
}
}
fp = ++bp;
}
}
printf ("set C_Include_Path=%s%s\n", fixeddirs, origdirs);
return 0;
}
/* Utility function that mallocs space and concatenates two strings. */
static char *
concat (s1, s2)
char *s1, *s2;
{
int len1 = strlen (s1);
int len2 = strlen (s2);
char *result = malloc (len1 + len2 + 1);
strcpy (result, s1);
strcpy (result + len1, s2);
*(result + len1 + len2) = 0;
return result;
}
/* Utility function that concatenates three strings. */
static char *
concat3 (s1, s2, s3)
char *s1, *s2, *s3;
{
return concat (concat (s1, s2), s3);
}
/* Utility function that concatenates four strings. */
static char *
concat4 (s1, s2, s3, s4)
char *s1, *s2, *s3, *s4;
{
return concat (concat (s1, s2), concat (s3, s4));
}
syntax highlighted by Code2HTML, v. 0.9.1