summaryrefslogtreecommitdiff
path: root/utils/vprojtomake/vcprojconvert.cpp
diff options
context:
space:
mode:
authorFluorescentCIAAfricanAmerican <[email protected]>2020-04-22 12:56:21 -0400
committerFluorescentCIAAfricanAmerican <[email protected]>2020-04-22 12:56:21 -0400
commit3bf9df6b2785fa6d951086978a3e66f49427166a (patch)
tree2c0f1f0c63c4832882bc93814ebd2c2b1c6224e5 /utils/vprojtomake/vcprojconvert.cpp
downloadarchived-source-engine-2018-hl2-src-master.tar.xz
archived-source-engine-2018-hl2-src-master.zip
Diffstat (limited to 'utils/vprojtomake/vcprojconvert.cpp')
-rw-r--r--utils/vprojtomake/vcprojconvert.cpp828
1 files changed, 828 insertions, 0 deletions
diff --git a/utils/vprojtomake/vcprojconvert.cpp b/utils/vprojtomake/vcprojconvert.cpp
new file mode 100644
index 0000000..df17b33
--- /dev/null
+++ b/utils/vprojtomake/vcprojconvert.cpp
@@ -0,0 +1,828 @@
+//========= Copyright Valve Corporation, All rights reserved. ============//
+//
+// Purpose:
+//
+// $NoKeywords: $
+//=============================================================================//
+#ifdef _LINUX
+#include <ctime> // needed by xercesc
+#endif
+
+#include "stdafx.h"
+#include "tier0/platform.h"
+#include <stdio.h>
+#include <stdlib.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+#ifdef _WIN32
+#include <windows.h>
+#include <comutil.h> // _variant_t
+#include <atlbase.h> // CComPtr
+#elif _LINUX
+#include <unistd.h>
+#include <dirent.h> // scandir()
+#define _stat stat
+
+
+#include <xercesc/util/PlatformUtils.hpp>
+#include <xercesc/util/XMLString.hpp>
+#include <xercesc/dom/DOM.hpp>
+#include <xercesc/sax/HandlerBase.hpp>
+#include <xercesc/parsers/XercesDOMParser.hpp>
+
+#include "valve_minmax_off.h"
+#if defined(XERCES_NEW_IOSTREAMS)
+#include <iostream>
+#else
+#include <iostream.h>
+#endif
+
+#include "valve_minmax_on.h"
+
+#define IXMLDOMNode DOMNode
+#define IXMLDOMNodeList DOMNodeList
+
+#define _alloca alloca
+
+XERCES_CPP_NAMESPACE_USE
+
+class XStr
+{
+public :
+ XStr(const char* const toTranscode)
+ {
+ // Call the private transcoding method
+ fUnicodeForm = XMLString::transcode(toTranscode);
+ }
+
+ ~XStr()
+ {
+ XMLString::release(&fUnicodeForm);
+ }
+
+
+ // -----------------------------------------------------------------------
+ // Getter methods
+ // -----------------------------------------------------------------------
+ const XMLCh* unicodeForm() const
+ {
+ return fUnicodeForm;
+ }
+
+private :
+ XMLCh* fUnicodeForm;
+};
+
+#define _bstr_t(str) XStr(str).unicodeForm()
+
+
+#else
+#error "Unsupported platform"
+#endif
+
+#include "vcprojconvert.h"
+#include "utlvector.h"
+
+//-----------------------------------------------------------------------------
+// Purpose: constructor
+//-----------------------------------------------------------------------------
+CVCProjConvert::CVCProjConvert()
+{
+#ifdef _WIN32
+ ::CoInitialize(NULL);
+#elif _LINUX
+ try {
+ XMLPlatformUtils::Initialize();
+ }
+ catch (const XMLException& toCatch) {
+ char* message = XMLString::transcode(toCatch.getMessage());
+ Error( "Error during initialization! : %s\n", message);
+ XMLString::release(&message);
+ }
+#endif
+ m_bProjectLoaded = false;
+}
+
+//-----------------------------------------------------------------------------
+// Purpose: destructor
+//-----------------------------------------------------------------------------
+CVCProjConvert::~CVCProjConvert()
+{
+#ifdef _WIN32
+ ::CoUninitialize();
+#elif _LINUX
+ // nothing to shutdown
+#endif
+}
+
+//-----------------------------------------------------------------------------
+// Purpose: load up a project and parse it
+//-----------------------------------------------------------------------------
+bool CVCProjConvert::LoadProject( const char *project )
+{
+#ifdef _WIN32
+ HRESULT hr;
+ IXMLDOMDocument *pXMLDoc=NULL;
+
+ hr = ::CoCreateInstance(CLSID_DOMDocument,
+ NULL,
+ CLSCTX_INPROC_SERVER,
+ IID_IXMLDOMDocument,
+ (void**)&pXMLDoc);
+
+ if (FAILED(hr))
+ {
+ Msg ("Cannot instantiate msxml2.dll\n");
+ Msg ("Please download the MSXML run-time (url below)\n");
+ Msg ("http://msdn.microsoft.com/downloads/default.asp?url=/downloads/sample.asp?url=/msdn-files/027/001/766/msdncompositedoc.xml\n");
+ return false;
+ }
+
+ VARIANT_BOOL vtbool;
+ _variant_t bstrProject(project);
+
+ pXMLDoc->put_async( VARIANT_BOOL(FALSE) );
+ hr = pXMLDoc->load(bstrProject,&vtbool);
+ if (FAILED(hr) || vtbool==VARIANT_FALSE)
+ {
+ Msg ("Could not open %s.\n", bstrProject);
+ pXMLDoc->Release();
+ return false;
+ }
+#elif _LINUX
+ XercesDOMParser* parser = new XercesDOMParser();
+ parser->setValidationScheme(XercesDOMParser::Val_Always); // optional.
+ parser->setDoNamespaces(true); // optional
+
+ ErrorHandler* errHandler = (ErrorHandler*) new HandlerBase();
+ parser->setErrorHandler(errHandler);
+
+ try {
+ parser->parse(project);
+ }
+ catch (const XMLException& toCatch) {
+ char* message = XMLString::transcode(toCatch.getMessage());
+ Error( "Exception message is: %s\n", message );
+ XMLString::release(&message);
+ return;
+ }
+ catch (const DOMException& toCatch) {
+ char* message = XMLString::transcode(toCatch.msg);
+ Error( "Exception message is: %s\n", message );
+ XMLString::release(&message);
+ return;
+ }
+ catch (...) {
+ Error( "Unexpected Exception \n" );
+ return;
+ }
+
+ DOMDocument *pXMLDoc = parser->getDocument();
+#endif
+
+ ExtractProjectName( pXMLDoc );
+ if ( !m_Name.IsValid() )
+ {
+ Msg( "Failed to extract project name\n" );
+ return false;
+ }
+ char baseDir[ MAX_PATH ];
+ Q_ExtractFilePath( project, baseDir, sizeof(baseDir) );
+ Q_StripTrailingSlash( baseDir );
+ m_BaseDir = baseDir;
+
+ ExtractConfigurations( pXMLDoc );
+ if ( m_Configurations.Count() == 0 )
+ {
+ Msg( "Failed to find any configurations to load\n" );
+ return false;
+ }
+
+ ExtractFiles( pXMLDoc );
+
+#ifdef _WIN32
+ pXMLDoc->Release();
+#elif _LINUX
+ delete pXMLDoc;
+ delete errHandler;
+#endif
+
+ m_bProjectLoaded = true;
+ return true;
+}
+
+
+//-----------------------------------------------------------------------------
+// Purpose: returns the number of different configurations loaded
+//-----------------------------------------------------------------------------
+int CVCProjConvert::GetNumConfigurations()
+{
+ Assert( m_bProjectLoaded );
+ return m_Configurations.Count();
+
+}
+
+//-----------------------------------------------------------------------------
+// Purpose: returns the index of a config with this name, -1 on err
+//-----------------------------------------------------------------------------
+int CVCProjConvert::FindConfiguration( CUtlSymbol name )
+{
+ if ( !name.IsValid() )
+ {
+ return -1;
+ }
+
+ for ( int i = 0; i < m_Configurations.Count(); i++ )
+ {
+ if ( m_Configurations[i].GetName() == name )
+ {
+ return i;
+ }
+ }
+ return -1;
+}
+
+//-----------------------------------------------------------------------------
+// Purpose: extracts the value of the xml attrib "attribName"
+//-----------------------------------------------------------------------------
+CUtlSymbol CVCProjConvert::GetXMLAttribValue( IXMLDOMElement *p, const char *attribName )
+{
+ if (!p)
+ {
+ return CUtlSymbol();
+ }
+
+#ifdef _WIN32
+ VARIANT vtValue;
+ p->getAttribute( _bstr_t(attribName), &vtValue);
+ if ( vtValue.vt == VT_NULL )
+ {
+ return CUtlSymbol(); // element not found
+ }
+
+ Assert( vtValue.vt == VT_BSTR );
+ CUtlSymbol name( static_cast<char *>( _bstr_t( vtValue.bstrVal ) ) );
+ ::SysFreeString(vtValue.bstrVal);
+#elif _LINUX
+ const XMLCh *xAttrib = XMLString::transcode( attribName );
+ const XMLCh *value = p->getAttribute( xAttrib );
+ if ( value == NULL )
+ {
+ return CUtlSymbol(); // element not found
+ }
+ char *transValue = XMLString::transcode(value);
+ CUtlSymbol name( transValue );
+ XMLString::release( &xAttrib );
+ XMLString::release( &transValue );
+#endif
+ return name;
+
+}
+
+//-----------------------------------------------------------------------------
+// Purpose: returns the name of this node
+//-----------------------------------------------------------------------------
+CUtlSymbol CVCProjConvert::GetXMLNodeName( IXMLDOMElement *p )
+{
+ CUtlSymbol name;
+ if (!p)
+ {
+ return name;
+ }
+
+#ifdef _WIN32
+ BSTR bstrName;
+ p->get_nodeName( &bstrName );
+ _bstr_t bstr(bstrName);
+ name = static_cast<char *>(bstr);
+ return name;
+#elif _LINUX
+ Assert( 0 );
+ Error( "Function CVCProjConvert::GetXMLNodeName not implemented\n" );
+ return name;
+#endif
+}
+
+//-----------------------------------------------------------------------------
+// Purpose: returns the config object at this index
+//-----------------------------------------------------------------------------
+CVCProjConvert::CConfiguration & CVCProjConvert::GetConfiguration( int i )
+{
+ Assert( m_bProjectLoaded );
+ Assert( m_Configurations.IsValidIndex(i) );
+ return m_Configurations[i];
+}
+
+//-----------------------------------------------------------------------------
+// Purpose: extracts the project name from the loaded vcproj
+//-----------------------------------------------------------------------------
+bool CVCProjConvert::ExtractProjectName( IXMLDOMDocument *pDoc )
+{
+#ifdef _WIN32
+ CComPtr<IXMLDOMNodeList> pProj;
+ pDoc->getElementsByTagName( _bstr_t("VisualStudioProject"), &pProj);
+ if (pProj)
+ {
+ long len = 0;
+ pProj->get_length(&len);
+ Assert( len == 1 );
+ if ( len == 1 )
+ {
+ CComPtr<IXMLDOMNode> pNode;
+ pProj->get_item( 0, &pNode );
+ if (pNode)
+ {
+ CComQIPtr<IXMLDOMElement> pElem( pNode );
+ m_Name = GetXMLAttribValue( pElem, "Name");
+ }
+ }
+ }
+#elif _LINUX
+ DOMNodeList *nodes = pDoc->getElementsByTagName( _bstr_t("VisualStudioProject") );
+ if ( nodes )
+ {
+ int len = nodes->getLength();
+ if ( len == 1 )
+ {
+ DOMNode *node = nodes->item(0);
+ if ( node )
+ {
+ m_Name = GetXMLAttribValue( node, "Name" );
+ }
+ }
+ }
+#endif
+ return true;
+}
+
+//-----------------------------------------------------------------------------
+// Purpose: extracts the list of configuration names from the vcproj
+//-----------------------------------------------------------------------------
+bool CVCProjConvert::ExtractConfigurations( IXMLDOMDocument *pDoc )
+{
+ m_Configurations.RemoveAll();
+
+ if (!pDoc)
+ {
+ return false;
+ }
+
+#ifdef _WIN32
+ CComPtr<IXMLDOMNodeList> pConfigs;
+ pDoc->getElementsByTagName( _bstr_t("Configuration"), &pConfigs);
+ if (pConfigs)
+ {
+ long len = 0;
+ pConfigs->get_length(&len);
+ for ( int i=0; i<len; i++ )
+ {
+ CComPtr<IXMLDOMNode> pNode;
+ pConfigs->get_item( i, &pNode );
+ if (pNode)
+ {
+ CComQIPtr<IXMLDOMElement> pElem( pNode );
+ CUtlSymbol configName = GetXMLAttribValue( pElem, "Name" );
+ if ( configName.IsValid() )
+ {
+ int newIndex = m_Configurations.AddToTail();
+ CConfiguration & config = m_Configurations[newIndex];
+ config.SetName( configName );
+ ExtractIncludes( pElem, config );
+ }
+ }
+ }
+ }
+#elif _LINUX
+ DOMNodeList *nodes = pDoc->getElementsByTagName( _bstr_t("Configuration"));
+ if ( nodes )
+ {
+ int len = nodes->getLength();
+ for ( int i=0; i<len; i++ )
+ {
+ DOMNode *node = nodes->item(i);
+ if (node)
+ {
+ CUtlSymbol configName = GetXMLAttribValue( node, "Name" );
+ if ( configName.IsValid() )
+ {
+ int newIndex = m_Configurations.AddToTail();
+ CConfiguration & config = m_Configurations[newIndex];
+ config.SetName( configName );
+ ExtractIncludes( node, config );
+ }
+ }
+ }
+ }
+#endif
+ return true;
+}
+
+//-----------------------------------------------------------------------------
+// Purpose: extracts the list of defines and includes used for this config
+//-----------------------------------------------------------------------------
+bool CVCProjConvert::ExtractIncludes( IXMLDOMElement *pDoc, CConfiguration & config )
+{
+ config.ResetDefines();
+ config.ResetIncludes();
+
+ if (!pDoc)
+ {
+ return false;
+ }
+
+#ifdef _WIN32
+ CComPtr<IXMLDOMNodeList> pTools;
+ pDoc->getElementsByTagName( _bstr_t("Tool"), &pTools);
+ if (pTools)
+ {
+ long len = 0;
+ pTools->get_length(&len);
+ for ( int i=0; i<len; i++ )
+ {
+ CComPtr<IXMLDOMNode> pNode;
+ pTools->get_item( i, &pNode );
+ if (pNode)
+ {
+ CComQIPtr<IXMLDOMElement> pElem( pNode );
+ CUtlSymbol toolName = GetXMLAttribValue( pElem, "Name" );
+ if ( toolName == "VCCLCompilerTool" )
+ {
+ CUtlSymbol defines = GetXMLAttribValue( pElem, "PreprocessorDefinitions" );
+ char *str = (char *)_alloca( Q_strlen( defines.String() ) + 1 );
+ Assert( str );
+ Q_strcpy( str, defines.String() );
+ // now tokenize the string on the ";" char
+ char *delim = strchr( str, ';' );
+ char *curpos = str;
+ while ( delim )
+ {
+ *delim = 0;
+ delim++;
+ if ( Q_stricmp( curpos, "WIN32" ) && Q_stricmp( curpos, "_WIN32" ) &&
+ Q_stricmp( curpos, "_WINDOWS") && Q_stricmp( curpos, "WINDOWS")) // don't add WIN32 defines
+ {
+ config.AddDefine( curpos );
+ }
+ curpos = delim;
+ delim = strchr( delim, ';' );
+ }
+ if ( Q_stricmp( curpos, "WIN32" ) && Q_stricmp( curpos, "_WIN32" ) &&
+ Q_stricmp( curpos, "_WINDOWS") && Q_stricmp( curpos, "WINDOWS")) // don't add WIN32 defines
+ {
+ config.AddDefine( curpos );
+ }
+
+ CUtlSymbol includes = GetXMLAttribValue( pElem, "AdditionalIncludeDirectories" );
+ char *str2 = (char *)_alloca( Q_strlen( includes.String() ) + 1 );
+ Assert( str2 );
+ Q_strcpy( str2, includes.String() );
+ // now tokenize the string on the ";" char
+ delim = strchr( str2, ',' );
+ curpos = str2;
+ while ( delim )
+ {
+ *delim = 0;
+ delim++;
+ config.AddInclude( curpos );
+ curpos = delim;
+ delim = strchr( delim, ',' );
+ }
+ config.AddInclude( curpos );
+ }
+ }
+ }
+ }
+#elif _LINUX
+ DOMNodeList *nodes= pDoc->getElementsByTagName( _bstr_t("Tool"));
+ if (nodes)
+ {
+ int len = nodes->getLength();
+ for ( int i=0; i<len; i++ )
+ {
+ DOMNode *node = nodes->item(i);
+ if (node)
+ {
+ CUtlSymbol toolName = GetXMLAttribValue( node, "Name" );
+ if ( toolName == "VCCLCompilerTool" )
+ {
+ CUtlSymbol defines = GetXMLAttribValue( node, "PreprocessorDefinitions" );
+ char *str = (char *)_alloca( Q_strlen( defines.String() ) + 1 );
+ Assert( str );
+ Q_strcpy( str, defines.String() );
+ // now tokenize the string on the ";" char
+ char *delim = strchr( str, ';' );
+ char *curpos = str;
+ while ( delim )
+ {
+ *delim = 0;
+ delim++;
+ if ( Q_stricmp( curpos, "WIN32" ) && Q_stricmp( curpos, "_WIN32" ) &&
+ Q_stricmp( curpos, "_WINDOWS") && Q_stricmp( curpos, "WINDOWS")) // don't add WIN32 defines
+ {
+ config.AddDefine( curpos );
+ }
+ curpos = delim;
+ delim = strchr( delim, ';' );
+ }
+ if ( Q_stricmp( curpos, "WIN32" ) && Q_stricmp( curpos, "_WIN32" ) &&
+ Q_stricmp( curpos, "_WINDOWS") && Q_stricmp( curpos, "WINDOWS")) // don't add WIN32 defines
+ {
+ config.AddDefine( curpos );
+ }
+
+ CUtlSymbol includes = GetXMLAttribValue( node, "AdditionalIncludeDirectories" );
+ char *str2 = (char *)_alloca( Q_strlen( includes.String() ) + 1 );
+ Assert( str2 );
+ Q_strcpy( str2, includes.String() );
+ // now tokenize the string on the ";" char
+ char token = ',';
+ delim = strchr( str2, token );
+ if ( !delim )
+ {
+ token = ';';
+ delim = strchr( str2, token );
+ }
+ curpos = str2;
+ while ( delim )
+ {
+ *delim = 0;
+ delim++;
+ Q_FixSlashes( curpos );
+ char fullPath[ MAX_PATH ];
+ Q_snprintf( fullPath, sizeof(fullPath), "%s/%s", m_BaseDir.String(), curpos );
+ Q_StripTrailingSlash( fullPath );
+ config.AddInclude( fullPath );
+ curpos = delim;
+ delim = strchr( delim, token );
+ }
+ Q_FixSlashes( curpos );
+ Q_strlower( curpos );
+ char fullPath[ MAX_PATH ];
+ Q_snprintf( fullPath, sizeof(fullPath), "%s/%s", m_BaseDir.String(), curpos );
+ Q_StripTrailingSlash( fullPath );
+ config.AddInclude( fullPath );
+ }
+ }
+ }
+ }
+
+#endif
+ return true;
+}
+
+
+//-----------------------------------------------------------------------------
+// Purpose: walks a particular files config entry and removes an files not valid for this config
+//-----------------------------------------------------------------------------
+bool CVCProjConvert::IterateFileConfigurations( IXMLDOMElement *pFile, CUtlSymbol fileName )
+{
+#ifdef _WIN32
+ CComPtr<IXMLDOMNodeList> pConfigs;
+ pFile->getElementsByTagName( _bstr_t("FileConfiguration"), &pConfigs);
+ if (pConfigs)
+ {
+ long len = 0;
+ pConfigs->get_length(&len);
+ for ( int i=0; i<len; i++ )
+ {
+ CComPtr<IXMLDOMNode> pNode;
+ pConfigs->get_item( i, &pNode);
+ if (pNode)
+ {
+ CComQIPtr<IXMLDOMElement> pElem( pNode );
+ CUtlSymbol configName = GetXMLAttribValue( pElem, "Name");
+ CUtlSymbol excluded = GetXMLAttribValue( pElem ,"ExcludedFromBuild");
+ if ( configName.IsValid() && excluded.IsValid() )
+ {
+ int index = FindConfiguration( configName );
+ if ( index > 0 && excluded == "TRUE" )
+ {
+ m_Configurations[index].RemoveFile( fileName );
+ }
+ }
+ }
+
+ }//for
+ }//if
+#elif _LINUX
+ DOMNodeList *nodes = pFile->getElementsByTagName( _bstr_t("FileConfiguration"));
+ if (nodes)
+ {
+ int len = nodes->getLength();
+ for ( int i=0; i<len; i++ )
+ {
+ DOMNode *node = nodes->item(i);
+ if (node)
+ {
+ CUtlSymbol configName = GetXMLAttribValue( node, "Name");
+ CUtlSymbol excluded = GetXMLAttribValue( node ,"ExcludedFromBuild");
+ if ( configName.IsValid() && excluded.IsValid() )
+ {
+ int index = FindConfiguration( configName );
+ if ( index >= 0 && excluded == "TRUE" )
+ {
+ m_Configurations[index].RemoveFile( fileName );
+ }
+ }
+ }
+
+ }//for
+ }//if
+#endif
+
+ return true;
+}
+
+//-----------------------------------------------------------------------------
+// Purpose: walks the file elements in the vcproj and inserts them into configs
+//-----------------------------------------------------------------------------
+bool CVCProjConvert::ExtractFiles( IXMLDOMDocument *pDoc )
+{
+ if (!pDoc)
+ {
+ return false;
+ }
+ Assert( m_Configurations.Count() ); // some configs must be loaded first
+
+#ifdef _WIN32
+ CComPtr<IXMLDOMNodeList> pFiles;
+ pDoc->getElementsByTagName( _bstr_t("File"), &pFiles);
+ if (pFiles)
+ {
+ long len = 0;
+ pFiles->get_length(&len);
+ for ( int i=0; i<len; i++ )
+ {
+ CComPtr<IXMLDOMNode> pNode;
+ pFiles->get_item( i, &pNode);
+ if (pNode)
+ {
+ CComQIPtr<IXMLDOMElement> pElem( pNode );
+ CUtlSymbol fileName = GetXMLAttribValue(pElem,"RelativePath");
+ if ( fileName.IsValid() )
+ {
+ CConfiguration::FileType_e type = GetFileType( fileName.String() );
+ CConfiguration::CFileEntry fileEntry( fileName.String(), type );
+ for ( int i = 0; i < m_Configurations.Count(); i++ ) // add the file to all configs
+ {
+ CConfiguration & config = m_Configurations[i];
+ config.InsertFile( fileEntry );
+ }
+ IterateFileConfigurations( pElem, fileName ); // now remove the excluded ones
+ }
+ }
+ }//for
+ }
+#elif _LINUX
+ DOMNodeList *nodes = pDoc->getElementsByTagName( _bstr_t("File") );
+ if (nodes)
+ {
+ int len = nodes->getLength();
+ for ( int i=0; i<len; i++ )
+ {
+ DOMNode *node = nodes->item(i);
+ if (node)
+ {
+ CUtlSymbol fileName = GetXMLAttribValue(node,"RelativePath");
+ if ( fileName.IsValid() )
+ {
+ char fixedFileName[ MAX_PATH ];
+ Q_strncpy( fixedFileName, fileName.String(), sizeof(fixedFileName) );
+ if ( fixedFileName[0] == '.' && fixedFileName[1] == '\\' )
+ {
+ Q_memmove( fixedFileName, fixedFileName+2, sizeof(fixedFileName)-2 );
+ }
+
+ Q_FixSlashes( fixedFileName );
+ FindFileCaseInsensitive( fixedFileName, sizeof(fixedFileName) );
+ CConfiguration::FileType_e type = GetFileType( fileName.String() );
+ CConfiguration::CFileEntry fileEntry( fixedFileName, type );
+ for ( int i = 0; i < m_Configurations.Count(); i++ ) // add the file to all configs
+ {
+ CConfiguration & config = m_Configurations[i];
+ config.InsertFile( fileEntry );
+ }
+ IterateFileConfigurations( node, fixedFileName ); // now remove the excluded ones
+ }
+ }
+ }//for
+ }
+#endif
+ return true;
+}
+
+#ifdef _LINUX
+static char fileName[MAX_PATH];
+int CheckName(const struct dirent *dir)
+{
+ return !strcasecmp( dir->d_name, fileName );
+}
+
+const char *findFileInDirCaseInsensitive(const char *file)
+{
+ const char *dirSep = strrchr(file,'/');
+ if( !dirSep )
+ {
+ dirSep=strrchr(file,'\\');
+ if( !dirSep )
+ {
+ return NULL;
+ }
+ }
+
+ char *dirName = static_cast<char *>( alloca( ( dirSep - file ) +1 ) );
+ if( !dirName )
+ return NULL;
+
+ Q_strncpy( dirName, file, dirSep - file );
+ dirName[ dirSep - file ] = '\0';
+
+ struct dirent **namelist;
+ int n;
+
+ Q_strncpy( fileName, dirSep + 1, MAX_PATH );
+
+
+ n = scandir( dirName , &namelist, CheckName, alphasort );
+
+ if( n > 0 )
+ {
+ while( n > 1 )
+ {
+ free( namelist[n] ); // free the malloc'd strings
+ n--;
+ }
+
+ Q_snprintf( fileName, sizeof( fileName ), "%s/%s", dirName, namelist[0]->d_name );
+ return fileName;
+ }
+ else
+ {
+ // last ditch attempt, just return the lower case version!
+ Q_strncpy( fileName, file, sizeof(fileName) );
+ Q_strlower( fileName );
+ return fileName;
+ }
+}
+#endif
+
+void CVCProjConvert::FindFileCaseInsensitive( char *fileName, int fileNameSize )
+{
+ char filePath[ MAX_PATH ];
+
+ Q_snprintf( filePath, sizeof(filePath), "%s/%s", m_BaseDir.String(), fileName );
+
+ struct _stat buf;
+ if ( _stat( filePath, &buf ) == 0)
+ {
+ return; // found the filename directly
+ }
+
+#ifdef _LINUX
+ const char *realName = findFileInDirCaseInsensitive( filePath );
+ if ( realName )
+ {
+ Q_strncpy( fileName, realName+strlen(m_BaseDir.String())+1, fileNameSize );
+ }
+#endif
+}
+
+//-----------------------------------------------------------------------------
+// Purpose: extracts the generic type of a file being loaded
+//-----------------------------------------------------------------------------
+CVCProjConvert::CConfiguration::FileType_e CVCProjConvert::GetFileType( const char *fileName )
+{
+ CConfiguration::FileType_e type = CConfiguration::FILE_TYPE_UNKNOWN_E;
+ char ext[10];
+ Q_ExtractFileExtension( fileName, ext, sizeof(ext) );
+ if ( !Q_stricmp( ext, "lib" ) )
+ {
+ type = CConfiguration::FILE_LIBRARY;
+ }
+ else if ( !Q_stricmp( ext, "h" ) )
+ {
+ type = CConfiguration::FILE_HEADER;
+ }
+ else if ( !Q_stricmp( ext, "hh" ) )
+ {
+ type = CConfiguration::FILE_HEADER;
+ }
+ else if ( !Q_stricmp( ext, "hpp" ) )
+ {
+ type = CConfiguration::FILE_HEADER;
+ }
+ else if ( !Q_stricmp( ext, "cpp" ) )
+ {
+ type = CConfiguration::FILE_SOURCE;
+ }
+ else if ( !Q_stricmp( ext, "c" ) )
+ {
+ type = CConfiguration::FILE_SOURCE;
+ }
+ else if ( !Q_stricmp( ext, "cc" ) )
+ {
+ type = CConfiguration::FILE_SOURCE;
+ }
+
+ return type;
+}