reactos/tools/rbuild/directory.cpp
Amine Khaldi 84291d0d72 * Bring back rbuild build to be used until bug 6372 is fixed.
svn path=/branches/GSoC_2011/ThemesSupport/; revision=52691
2011-07-15 22:27:16 +00:00

250 lines
5.6 KiB
C++

/*
* Copyright (C) 2005 Casper S. Hornstrup
*
* This program 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.
*
* This program 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 this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#include "pch.h"
#include <assert.h>
#include "rbuild.h"
#ifdef _MSC_VER
#include <errno.h>
#else
#include <dirent.h>
#endif//_MSC_VER
#ifdef WIN32
#define MKDIR(s) mkdir(s)
#else
#define MKDIR(s) mkdir(s, 0755)
#endif
using std::string;
using std::vector;
Directory::Directory ( const string& name_ )
: name(name_)
{
size_t pos = name.find_first_of ( "$:" );
if ( pos != string::npos )
{
throw InvalidOperationException ( __FILE__,
__LINE__,
"Invalid directory name '%s'",
name.c_str() );
}
const char* p = strpbrk ( name_.c_str (), "/\\" );
if ( name_.c_str () == p )
{
throw InvalidOperationException ( __FILE__,
__LINE__,
"Invalid relative path '%s'",
name_.c_str () );
}
if ( p )
{
name.erase ( p - name_.c_str ());
Add ( p + 1 );
}
}
void
Directory::Add ( const char* subdir )
{
size_t i;
string s1 = string ( subdir );
if ( ( i = s1.find ( '$' ) ) != string::npos )
{
throw InvalidOperationException ( __FILE__,
__LINE__,
"No environment variables can be used here. Path was %s",
subdir );
}
const char* p = strpbrk ( subdir, "/\\" );
if ( subdir == p || ( *subdir && subdir[1] == ':' ) )
{
throw InvalidOperationException ( __FILE__,
__LINE__,
"Invalid relative path '%s'",
subdir );
}
if ( !p )
p = subdir + strlen(subdir);
string s ( subdir, p-subdir );
if ( subdirs.find(s) == subdirs.end() )
subdirs[s] = new Directory(s);
if ( *p && *++p )
subdirs[s]->Add ( p );
}
bool
Directory::mkdir_p ( const char* path )
{
#ifndef _MSC_VER
DIR *directory;
directory = opendir ( path );
if ( directory != NULL )
{
closedir ( directory );
return false;
}
#endif//_MSC_VER
if ( MKDIR ( path ) != 0 )
{
#ifdef _MSC_VER
if ( errno == EEXIST )
return false;
#endif//_MSC_VER
throw AccessDeniedException ( string ( path ) );
}
return true;
}
bool
Directory::CreateDirectory ( const string& path )
{
size_t index = 0;
size_t nextIndex;
if ( isalpha ( path[0] ) && path[1] == ':' && path[2] == cSep )
{
nextIndex = path.find ( cSep, 3);
index = path.find ( cSep );
}
else
nextIndex = path.find ( cSep );
bool directoryWasCreated = false;
while ( nextIndex != string::npos )
{
nextIndex = path.find ( cSep, index + 1 );
directoryWasCreated = mkdir_p ( path.substr ( 0, nextIndex ).c_str () );
index = nextIndex;
}
return directoryWasCreated;
}
void
Directory::GenerateTree ( DirectoryLocation root,
bool verbose )
{
switch ( root )
{
case IntermediateDirectory:
return GenerateTree ( Environment::GetIntermediatePath (), verbose );
case OutputDirectory:
return GenerateTree ( Environment::GetOutputPath (), verbose );
case InstallDirectory:
return GenerateTree ( Environment::GetInstallPath (), verbose );
default:
throw InvalidOperationException ( __FILE__,
__LINE__,
"Invalid directory %d.",
root );
}
}
void
Directory::GenerateTree ( const string& parent,
bool verbose )
{
string path;
if ( parent.size () > 0 )
{
if ( name.size () > 0 )
path = parent + sSep + name;
else
path = parent;
if ( CreateDirectory ( path ) && verbose )
printf ( "Created %s\n", path.c_str () );
}
else
path = name;
for ( directory_map::iterator i = subdirs.begin ();
i != subdirs.end ();
++i )
{
i->second->GenerateTree ( path, verbose );
}
}
string
Directory::EscapeSpaces ( const string& path )
{
string newpath;
const char* p = &path[0];
while ( *p != 0 )
{
if ( *p == ' ' )
newpath = newpath + "\\ ";
else
newpath = newpath + *p;
p++;
}
return newpath;
}
void
Directory::CreateRule ( FILE* f,
const string& parent )
{
string path;
string escapedName = EscapeSpaces ( name );
if ( escapedName.size() > 0 )
{
if ( ! (escapedName == "tools" &&
( parent == "$(OUTPUT)" || parent == "$(INTERMEDIATE)" ) ) )
{
fprintf ( f,
"%s%c%s: | %s\n",
parent.c_str (),
cSep,
escapedName.c_str (),
parent.c_str () );
fprintf ( f,
"\t$(ECHO_MKDIR)\n" );
fprintf ( f,
"\t${mkdir} $@\n" );
}
path = parent + sSep + escapedName;
}
else
path = parent;
for ( directory_map::iterator i = subdirs.begin();
i != subdirs.end();
++i )
{
i->second->CreateRule ( f, path );
}
}
Directory::~Directory()
{
std::map<std::string, Directory*>::iterator theIterator;
for ( theIterator = subdirs.begin (); theIterator != subdirs.end (); theIterator++ )
delete theIterator->second;
}