Blame qtools/qdir_unix.cpp

Packit 1c1d7e
/****************************************************************************
Packit 1c1d7e
** 
Packit 1c1d7e
**
Packit 1c1d7e
** Implementation of QDirclass
Packit 1c1d7e
**
Packit 1c1d7e
** Created : 950628
Packit 1c1d7e
**
Packit 1c1d7e
** Copyright (C) 1992-2000 Trolltech AS.  All rights reserved.
Packit 1c1d7e
**
Packit 1c1d7e
** This file is part of the tools module of the Qt GUI Toolkit.
Packit 1c1d7e
**
Packit 1c1d7e
** This file may be distributed under the terms of the Q Public License
Packit 1c1d7e
** as defined by Trolltech AS of Norway and appearing in the file
Packit 1c1d7e
** LICENSE.QPL included in the packaging of this file.
Packit 1c1d7e
**
Packit 1c1d7e
** This file may be distributed and/or modified under the terms of the
Packit 1c1d7e
** GNU General Public License version 2 as published by the Free Software
Packit 1c1d7e
** Foundation and appearing in the file LICENSE.GPL included in the
Packit 1c1d7e
** packaging of this file.
Packit 1c1d7e
**
Packit 1c1d7e
** Licensees holding valid Qt Enterprise Edition or Qt Professional Edition
Packit 1c1d7e
** licenses for Unix/X11 or for Qt/Embedded may use this file in accordance
Packit 1c1d7e
** with the Qt Commercial License Agreement provided with the Software.
Packit 1c1d7e
**
Packit 1c1d7e
** This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE
Packit 1c1d7e
** WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
Packit 1c1d7e
**
Packit 1c1d7e
** See http://www.trolltech.com/pricing.html or email sales@trolltech.com for
Packit 1c1d7e
**   information about Qt Commercial License Agreements.
Packit 1c1d7e
** See http://www.trolltech.com/qpl/ for QPL licensing information.
Packit 1c1d7e
** See http://www.trolltech.com/gpl/ for GPL licensing information.
Packit 1c1d7e
**
Packit 1c1d7e
** Contact info@trolltech.com if any conditions of this licensing are
Packit 1c1d7e
** not clear to you.
Packit 1c1d7e
**
Packit 1c1d7e
**********************************************************************/
Packit 1c1d7e
Packit 1c1d7e
#include "qglobal.h"
Packit 1c1d7e
Packit 1c1d7e
#include "qdir.h"
Packit 1c1d7e
#ifndef QT_NO_DIR
Packit 1c1d7e
Packit 1c1d7e
#include "qfileinfo.h"
Packit 1c1d7e
#include "qfiledefs_p.h"
Packit 1c1d7e
#include "qregexp.h"
Packit 1c1d7e
#include "qstringlist.h"
Packit 1c1d7e
#include <stdlib.h>
Packit 1c1d7e
#include <ctype.h>
Packit 1c1d7e
Packit 1c1d7e
extern QStringList qt_makeFilterList( const QString &filter );
Packit 1c1d7e
Packit 1c1d7e
extern int qt_cmp_si_sortSpec;
Packit 1c1d7e
Packit 1c1d7e
#if defined(Q_C_CALLBACKS)
Packit 1c1d7e
extern "C" {
Packit 1c1d7e
#endif
Packit 1c1d7e
Packit 1c1d7e
extern int qt_cmp_si( const void *, const void * );
Packit 1c1d7e
Packit 1c1d7e
#if defined(Q_C_CALLBACKS)
Packit 1c1d7e
}
Packit 1c1d7e
#endif
Packit 1c1d7e
Packit 1c1d7e
Packit 1c1d7e
void QDir::slashify( QString& )
Packit 1c1d7e
{
Packit 1c1d7e
}
Packit 1c1d7e
Packit 1c1d7e
QString QDir::homeDirPath()
Packit 1c1d7e
{
Packit 1c1d7e
    QString d;
Packit 1c1d7e
    d = QFile::decodeName(getenv("HOME"));
Packit 1c1d7e
    slashify( d );
Packit 1c1d7e
    if ( d.isNull() )
Packit 1c1d7e
	d = rootDirPath();
Packit 1c1d7e
    return d;
Packit 1c1d7e
}
Packit 1c1d7e
Packit 1c1d7e
QString QDir::canonicalPath() const
Packit 1c1d7e
{
Packit 1c1d7e
    QString r;
Packit 1c1d7e
Packit 1c1d7e
    char cur[PATH_MAX];
Packit 1c1d7e
    char tmp[PATH_MAX];
Packit 1c1d7e
    if (GETCWD( cur, PATH_MAX )) {
Packit 1c1d7e
      if ( CHDIR(QFile::encodeName(dPath)) >= 0 ) {
Packit 1c1d7e
  	if (GETCWD( tmp, PATH_MAX )) {
Packit 1c1d7e
  	  r = QFile::decodeName(tmp);
Packit 1c1d7e
        }
Packit 1c1d7e
        (void)CHDIR( cur );
Packit 1c1d7e
      }
Packit 1c1d7e
    }
Packit 1c1d7e
Packit 1c1d7e
    slashify( r );
Packit 1c1d7e
    return r;
Packit 1c1d7e
}
Packit 1c1d7e
Packit 1c1d7e
bool QDir::mkdir( const QString &dirName, bool acceptAbsPath ) const
Packit 1c1d7e
{
Packit 1c1d7e
    return MKDIR( QFile::encodeName(filePath(dirName,acceptAbsPath)), 0777 ) 
Packit 1c1d7e
	== 0;
Packit 1c1d7e
}
Packit 1c1d7e
Packit 1c1d7e
bool QDir::rmdir( const QString &dirName, bool acceptAbsPath ) const
Packit 1c1d7e
{
Packit 1c1d7e
    return RMDIR( QFile::encodeName(filePath(dirName,acceptAbsPath)) ) == 0;
Packit 1c1d7e
}
Packit 1c1d7e
Packit 1c1d7e
bool QDir::isReadable() const
Packit 1c1d7e
{
Packit 1c1d7e
    return ACCESS( QFile::encodeName(dPath), R_OK | X_OK ) == 0;
Packit 1c1d7e
}
Packit 1c1d7e
Packit 1c1d7e
bool QDir::isRoot() const
Packit 1c1d7e
{
Packit 1c1d7e
    return dPath == QString::fromLatin1("/");
Packit 1c1d7e
}
Packit 1c1d7e
Packit 1c1d7e
bool QDir::rename( const QString &name, const QString &newName,
Packit 1c1d7e
		   bool acceptAbsPaths	)
Packit 1c1d7e
{
Packit 1c1d7e
    if ( name.isEmpty() || newName.isEmpty() ) {
Packit 1c1d7e
#if defined(CHECK_NULL)
Packit 1c1d7e
	qWarning( "QDir::rename: Empty or null file name(s)" );
Packit 1c1d7e
#endif
Packit 1c1d7e
	return FALSE;
Packit 1c1d7e
    }
Packit 1c1d7e
    QString fn1 = filePath( name, acceptAbsPaths );
Packit 1c1d7e
    QString fn2 = filePath( newName, acceptAbsPaths );
Packit 1c1d7e
    return ::rename( QFile::encodeName(fn1),
Packit 1c1d7e
		     QFile::encodeName(fn2) ) == 0;
Packit 1c1d7e
}
Packit 1c1d7e
Packit 1c1d7e
bool QDir::setCurrent( const QString &path )
Packit 1c1d7e
{
Packit 1c1d7e
    int r;
Packit 1c1d7e
    r = CHDIR( QFile::encodeName(path) );
Packit 1c1d7e
    return r >= 0;
Packit 1c1d7e
}
Packit 1c1d7e
Packit 1c1d7e
QString QDir::currentDirPath()
Packit 1c1d7e
{
Packit 1c1d7e
    QString result;
Packit 1c1d7e
Packit 1c1d7e
    STATBUF st;
Packit 1c1d7e
    if ( STAT( ".", &st ) == 0 ) {
Packit 1c1d7e
	char currentName[PATH_MAX];
Packit 1c1d7e
	if ( GETCWD( currentName, PATH_MAX ) != 0 )
Packit 1c1d7e
	    result = QFile::decodeName(currentName);
Packit 1c1d7e
#if defined(DEBUG)
Packit 1c1d7e
	if ( result.isNull() )
Packit 1c1d7e
	    qWarning( "QDir::currentDirPath: getcwd() failed" );
Packit 1c1d7e
#endif
Packit 1c1d7e
    } else {
Packit 1c1d7e
#if defined(DEBUG)
Packit 1c1d7e
	qWarning( "QDir::currentDirPath: stat(\".\") failed" );
Packit 1c1d7e
#endif
Packit 1c1d7e
    }
Packit 1c1d7e
    slashify( result );
Packit 1c1d7e
    return result;
Packit 1c1d7e
}
Packit 1c1d7e
Packit 1c1d7e
QString QDir::rootDirPath()
Packit 1c1d7e
{
Packit 1c1d7e
    QString d = QString::fromLatin1( "/" );
Packit 1c1d7e
    return d;
Packit 1c1d7e
}
Packit 1c1d7e
Packit 1c1d7e
bool QDir::isRelativePath( const QString &path )
Packit 1c1d7e
{
Packit 1c1d7e
    int len = path.length();
Packit 1c1d7e
    if ( len == 0 )
Packit 1c1d7e
	return TRUE;
Packit 1c1d7e
    return path[0] != '/';
Packit 1c1d7e
}
Packit 1c1d7e
Packit 1c1d7e
bool QDir::readDirEntries( const QString &nameFilter,
Packit 1c1d7e
			   int filterSpec, int sortSpec )
Packit 1c1d7e
{
Packit 1c1d7e
    int i;
Packit 1c1d7e
    if ( !fList ) {
Packit 1c1d7e
	fList  = new QStringList;
Packit 1c1d7e
	CHECK_PTR( fList );
Packit 1c1d7e
	fiList = new QFileInfoList;
Packit 1c1d7e
	CHECK_PTR( fiList );
Packit 1c1d7e
	fiList->setAutoDelete( TRUE );
Packit 1c1d7e
    } else {
Packit 1c1d7e
	fList->clear();
Packit 1c1d7e
	fiList->clear();
Packit 1c1d7e
    }
Packit 1c1d7e
Packit 1c1d7e
    QStringList filters = qt_makeFilterList( nameFilter );
Packit 1c1d7e
Packit 1c1d7e
    bool doDirs	    = (filterSpec & Dirs)	!= 0;
Packit 1c1d7e
    bool doFiles    = (filterSpec & Files)	!= 0;
Packit 1c1d7e
    bool noSymLinks = (filterSpec & NoSymLinks) != 0;
Packit 1c1d7e
    bool doReadable = (filterSpec & Readable)	!= 0;
Packit 1c1d7e
    bool doWritable = (filterSpec & Writable)	!= 0;
Packit 1c1d7e
    bool doExecable = (filterSpec & Executable) != 0;
Packit 1c1d7e
    bool doHidden   = (filterSpec & Hidden)	!= 0;
Packit 1c1d7e
Packit 1c1d7e
#if defined(_OS_OS2EMX_)
Packit 1c1d7e
    //QRegExp   wc( nameFilter, FALSE, TRUE );	// wild card, case insensitive
Packit 1c1d7e
#else
Packit 1c1d7e
    //QRegExp   wc( nameFilter, TRUE, TRUE );	// wild card, case sensitive
Packit 1c1d7e
#endif
Packit 1c1d7e
    QFileInfo fi;
Packit 1c1d7e
    DIR	     *dir;
Packit 1c1d7e
    dirent   *file;
Packit 1c1d7e
Packit 1c1d7e
    dir = opendir( QFile::encodeName(dPath) );
Packit 1c1d7e
    if ( !dir ) {
Packit 1c1d7e
#if defined(CHECK_NULL)
Packit 1c1d7e
	qWarning( "QDir::readDirEntries: Cannot read the directory: %s",
Packit 1c1d7e
		  QFile::encodeName(dPath).data() );
Packit 1c1d7e
#endif
Packit 1c1d7e
	return FALSE;
Packit 1c1d7e
    }
Packit 1c1d7e
Packit 1c1d7e
    while ( (file = readdir(dir)) ) {
Packit 1c1d7e
	QString fn = QFile::decodeName(file->d_name);
Packit 1c1d7e
	fi.setFile( *this, fn );
Packit 1c1d7e
	if ( !match( filters, fn ) && !(allDirs && fi.isDir()) )
Packit 1c1d7e
	     continue;
Packit 1c1d7e
	if  ( (doDirs && fi.isDir()) || (doFiles && fi.isFile()) ) {
Packit 1c1d7e
	    if ( noSymLinks && fi.isSymLink() )
Packit 1c1d7e
	        continue;
Packit 1c1d7e
	    if ( (filterSpec & RWEMask) != 0 )
Packit 1c1d7e
	        if ( (doReadable && !fi.isReadable()) ||
Packit 1c1d7e
	             (doWritable && !fi.isWritable()) ||
Packit 1c1d7e
	             (doExecable && !fi.isExecutable()) )
Packit 1c1d7e
	            continue;
Packit 1c1d7e
	    if ( !doHidden && fn[0] == '.' &&
Packit 1c1d7e
	         fn != QString::fromLatin1(".")
Packit 1c1d7e
	         && fn != QString::fromLatin1("..") )
Packit 1c1d7e
	        continue;
Packit 1c1d7e
	    fiList->append( new QFileInfo( fi ) );
Packit 1c1d7e
	}
Packit 1c1d7e
    }
Packit 1c1d7e
    if ( closedir(dir) != 0 ) {
Packit 1c1d7e
#if defined(CHECK_NULL)
Packit 1c1d7e
	qWarning( "QDir::readDirEntries: Cannot close the directory: %s",
Packit 1c1d7e
		  dPath.local8Bit().data() );
Packit 1c1d7e
#endif
Packit 1c1d7e
    }
Packit 1c1d7e
Packit 1c1d7e
    // Sort...
Packit 1c1d7e
    if(fiList->count()) {
Packit 1c1d7e
	QDirSortItem* si= new QDirSortItem[fiList->count()];
Packit 1c1d7e
	QFileInfo* itm;
Packit 1c1d7e
	i=0;
Packit 1c1d7e
	for (itm = fiList->first(); itm; itm = fiList->next())
Packit 1c1d7e
	    si[i++].item = itm;
Packit 1c1d7e
	qt_cmp_si_sortSpec = sortSpec;
Packit 1c1d7e
	qsort( si, i, sizeof(si[0]), qt_cmp_si );
Packit 1c1d7e
	// put them back in the list
Packit 1c1d7e
	fiList->setAutoDelete( FALSE );
Packit 1c1d7e
	fiList->clear();
Packit 1c1d7e
	int j;
Packit 1c1d7e
	for ( j=0; j
Packit 1c1d7e
	    fiList->append( si[j].item );
Packit 1c1d7e
	    fList->append( si[j].item->fileName() );
Packit 1c1d7e
	}
Packit 1c1d7e
	delete [] si;
Packit 1c1d7e
	fiList->setAutoDelete( TRUE );
Packit 1c1d7e
    }
Packit 1c1d7e
Packit 1c1d7e
    if ( filterSpec == (FilterSpec)filtS && sortSpec == (SortSpec)sortS &&
Packit 1c1d7e
	 nameFilter == nameFilt )
Packit 1c1d7e
	dirty = FALSE;
Packit 1c1d7e
    else
Packit 1c1d7e
	dirty = TRUE;
Packit 1c1d7e
    return TRUE;
Packit 1c1d7e
}
Packit 1c1d7e
Packit 1c1d7e
const QFileInfoList * QDir::drives()
Packit 1c1d7e
{
Packit 1c1d7e
    // at most one instance of QFileInfoList is leaked, and this variable
Packit 1c1d7e
    // points to that list
Packit 1c1d7e
    static QFileInfoList * knownMemoryLeak = 0;
Packit 1c1d7e
Packit 1c1d7e
    if ( !knownMemoryLeak ) {
Packit 1c1d7e
	knownMemoryLeak = new QFileInfoList;
Packit 1c1d7e
	// non-win32 versions both use just one root directory
Packit 1c1d7e
	knownMemoryLeak->append( new QFileInfo( rootDirPath() ) );
Packit 1c1d7e
    }
Packit 1c1d7e
Packit 1c1d7e
    return knownMemoryLeak;
Packit 1c1d7e
}
Packit 1c1d7e
#endif //QT_NO_DIR