2007-09-14 22:16:22 +00:00
|
|
|
/***************************************************************************
|
|
|
|
* __________ __ ___.
|
|
|
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
|
|
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
|
|
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
|
|
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
|
|
|
* \/ \/ \/ \/ \/
|
|
|
|
*
|
|
|
|
* Copyright (C) 2007 by Dominik Wenger
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
* All files in this archive are subject to the GNU General Public License.
|
|
|
|
* See the file COPYING in the source tree root for full license agreement.
|
|
|
|
*
|
|
|
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
|
|
|
* KIND, either express or implied.
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include "utils.h"
|
2010-04-02 21:24:19 +00:00
|
|
|
#include "rockboxinfo.h"
|
2009-08-10 19:46:51 +00:00
|
|
|
#include "system.h"
|
|
|
|
#include "rbsettings.h"
|
2010-01-24 21:12:11 +00:00
|
|
|
#include "systeminfo.h"
|
2009-08-10 19:46:51 +00:00
|
|
|
|
2008-06-29 07:45:33 +00:00
|
|
|
#ifdef UNICODE
|
|
|
|
#define _UNICODE
|
|
|
|
#endif
|
2007-09-14 22:16:22 +00:00
|
|
|
|
2008-06-21 10:28:10 +00:00
|
|
|
#include <QtCore>
|
|
|
|
#include <QDebug>
|
2008-03-18 23:22:08 +00:00
|
|
|
#include <cstdlib>
|
2008-06-21 10:28:10 +00:00
|
|
|
#include <stdio.h>
|
2007-09-14 22:16:22 +00:00
|
|
|
|
2008-06-29 07:45:33 +00:00
|
|
|
#if defined(Q_OS_WIN32)
|
|
|
|
#include <windows.h>
|
|
|
|
#include <tchar.h>
|
|
|
|
#include <winioctl.h>
|
|
|
|
#endif
|
2009-01-07 13:31:19 +00:00
|
|
|
#if defined(Q_OS_LINUX) || defined(Q_OS_MACX)
|
2008-12-13 20:09:31 +00:00
|
|
|
#include <sys/statvfs.h>
|
|
|
|
#endif
|
2008-06-29 07:45:33 +00:00
|
|
|
|
2007-09-14 22:16:22 +00:00
|
|
|
// recursive function to delete a dir with files
|
2010-04-02 21:24:19 +00:00
|
|
|
bool Utils::recursiveRmdir( const QString &dirName )
|
2007-09-14 22:16:22 +00:00
|
|
|
{
|
2007-10-12 16:54:51 +00:00
|
|
|
QString dirN = dirName;
|
|
|
|
QDir dir(dirN);
|
|
|
|
// make list of entries in directory
|
|
|
|
QStringList list = dir.entryList(QDir::AllEntries | QDir::NoDotAndDotDot);
|
|
|
|
QFileInfo fileInfo;
|
|
|
|
QString curItem, lstAt;
|
|
|
|
for(int i = 0; i < list.size(); i++){ // loop through all items of list
|
|
|
|
QString name = list.at(i);
|
|
|
|
curItem = dirN + "/" + name;
|
|
|
|
fileInfo.setFile(curItem);
|
|
|
|
if(fileInfo.isDir()) // is directory
|
2010-04-02 21:24:19 +00:00
|
|
|
recursiveRmdir(curItem); // call recRmdir() recursively for
|
|
|
|
// deleting subdirectory
|
2007-10-12 16:54:51 +00:00
|
|
|
else // is file
|
2010-04-02 21:24:19 +00:00
|
|
|
QFile::remove(curItem); // ok, delete file
|
2007-09-14 22:16:22 +00:00
|
|
|
}
|
2007-10-12 16:54:51 +00:00
|
|
|
dir.cdUp();
|
2010-04-02 21:24:19 +00:00
|
|
|
return dir.rmdir(dirN); // delete empty dir and return if (now empty)
|
|
|
|
// dir-removing was successfull
|
2007-09-14 22:16:22 +00:00
|
|
|
}
|
2008-01-23 21:54:40 +00:00
|
|
|
|
|
|
|
|
2008-04-01 20:30:41 +00:00
|
|
|
//! @brief resolves the given path, ignoring case.
|
|
|
|
//! @param path absolute path to resolve.
|
|
|
|
//! @return returns exact casing of path, empty string if path not found.
|
2010-04-02 21:24:19 +00:00
|
|
|
QString Utils::resolvePathCase(QString path)
|
2008-04-01 20:30:41 +00:00
|
|
|
{
|
|
|
|
QStringList elems;
|
2008-04-05 23:49:23 +00:00
|
|
|
QString realpath;
|
|
|
|
|
2008-04-01 20:30:41 +00:00
|
|
|
elems = path.split("/", QString::SkipEmptyParts);
|
2008-04-05 23:49:23 +00:00
|
|
|
int start;
|
|
|
|
#if defined(Q_OS_WIN32)
|
|
|
|
// on windows we must make sure to start with the first entry (i.e. the
|
|
|
|
// drive letter) instead of a single / to make resolving work.
|
|
|
|
start = 1;
|
|
|
|
realpath = elems.at(0) + "/";
|
|
|
|
#else
|
|
|
|
start = 0;
|
|
|
|
realpath = "/";
|
|
|
|
#endif
|
2008-04-01 20:30:41 +00:00
|
|
|
|
2008-04-05 23:49:23 +00:00
|
|
|
for(int i = start; i < elems.size(); i++) {
|
2008-04-06 19:50:24 +00:00
|
|
|
QStringList direlems
|
2008-12-13 20:09:31 +00:00
|
|
|
= QDir(realpath).entryList(QDir::AllEntries|QDir::Hidden|QDir::System);
|
2008-04-01 20:30:41 +00:00
|
|
|
if(direlems.contains(elems.at(i), Qt::CaseInsensitive)) {
|
|
|
|
// need to filter using QRegExp as QStringList::filter(QString)
|
|
|
|
// matches any substring
|
|
|
|
QString expr = QString("^" + elems.at(i) + "$");
|
|
|
|
QRegExp rx = QRegExp(expr, Qt::CaseInsensitive);
|
|
|
|
QStringList a = direlems.filter(rx);
|
|
|
|
|
|
|
|
if(a.size() != 1)
|
|
|
|
return QString("");
|
|
|
|
if(!realpath.endsWith("/"))
|
|
|
|
realpath += "/";
|
|
|
|
realpath += a.at(0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return QString("");
|
|
|
|
}
|
2009-10-29 19:43:48 +00:00
|
|
|
qDebug() << "[Utils] resolving path" << path << "->" << realpath;
|
2008-04-01 20:30:41 +00:00
|
|
|
return realpath;
|
|
|
|
}
|
|
|
|
|
2008-12-13 20:09:31 +00:00
|
|
|
|
|
|
|
//! @brief figure the free disk space on a filesystem
|
|
|
|
//! @param path path on the filesystem to check
|
|
|
|
//! @return size in bytes
|
2010-04-02 21:24:19 +00:00
|
|
|
qulonglong Utils::filesystemFree(QString path)
|
2011-07-15 18:13:31 +00:00
|
|
|
{
|
|
|
|
return filesystemSize(path, FilesystemFree);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
qulonglong Utils::filesystemTotal(QString path)
|
|
|
|
{
|
|
|
|
return filesystemSize(path, FilesystemTotal);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
qulonglong Utils::filesystemSize(QString path, enum Utils::Size type)
|
2008-12-13 20:09:31 +00:00
|
|
|
{
|
|
|
|
qlonglong size = 0;
|
2009-01-07 13:31:19 +00:00
|
|
|
#if defined(Q_OS_LINUX) || defined(Q_OS_MACX)
|
2008-12-13 20:09:31 +00:00
|
|
|
// the usage of statfs() is deprecated by the LSB so use statvfs().
|
|
|
|
struct statvfs fs;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = statvfs(qPrintable(path), &fs);
|
|
|
|
|
2011-07-15 18:13:31 +00:00
|
|
|
if(ret == 0) {
|
|
|
|
if(type == FilesystemFree) {
|
|
|
|
size = (qulonglong)fs.f_frsize * (qulonglong)fs.f_bavail;
|
|
|
|
}
|
|
|
|
if(type == FilesystemTotal) {
|
|
|
|
size = (qulonglong)fs.f_frsize * (qulonglong)fs.f_blocks;
|
|
|
|
}
|
|
|
|
}
|
2008-12-13 20:09:31 +00:00
|
|
|
#endif
|
|
|
|
#if defined(Q_OS_WIN32)
|
|
|
|
BOOL ret;
|
|
|
|
ULARGE_INTEGER freeAvailBytes;
|
2011-07-15 18:13:31 +00:00
|
|
|
ULARGE_INTEGER totalNumberBytes;
|
2008-12-13 20:09:31 +00:00
|
|
|
|
2011-07-15 18:13:31 +00:00
|
|
|
ret = GetDiskFreeSpaceExW((LPCTSTR)path.utf16(), &freeAvailBytes,
|
|
|
|
&totalNumberBytes, NULL);
|
|
|
|
if(ret) {
|
|
|
|
if(type == FilesystemFree) {
|
|
|
|
size = freeAvailBytes.QuadPart;
|
|
|
|
}
|
|
|
|
if(type == FilesystemTotal) {
|
|
|
|
size = totalNumberBytes.QuadPart;
|
|
|
|
}
|
|
|
|
}
|
2008-12-13 20:09:31 +00:00
|
|
|
#endif
|
2010-06-18 19:46:45 +00:00
|
|
|
qDebug() << "[Utils] Filesystem free:" << path << size;
|
2008-12-13 20:09:31 +00:00
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
2009-04-29 21:27:01 +00:00
|
|
|
//! \brief searches for a Executable in the Environement Path
|
2010-04-02 21:24:19 +00:00
|
|
|
QString Utils::findExecutable(QString name)
|
2009-04-29 21:27:01 +00:00
|
|
|
{
|
|
|
|
QString exepath;
|
|
|
|
//try autodetect tts
|
|
|
|
#if defined(Q_OS_LINUX) || defined(Q_OS_MACX) || defined(Q_OS_OPENBSD)
|
|
|
|
QStringList path = QString(getenv("PATH")).split(":", QString::SkipEmptyParts);
|
|
|
|
#elif defined(Q_OS_WIN)
|
|
|
|
QStringList path = QString(getenv("PATH")).split(";", QString::SkipEmptyParts);
|
|
|
|
#endif
|
2009-10-29 19:43:48 +00:00
|
|
|
qDebug() << "[Utils] system path:" << path;
|
2009-04-29 21:27:01 +00:00
|
|
|
for(int i = 0; i < path.size(); i++)
|
|
|
|
{
|
|
|
|
QString executable = QDir::fromNativeSeparators(path.at(i)) + "/" + name;
|
|
|
|
#if defined(Q_OS_WIN)
|
|
|
|
executable += ".exe";
|
|
|
|
QStringList ex = executable.split("\"", QString::SkipEmptyParts);
|
|
|
|
executable = ex.join("");
|
|
|
|
#endif
|
2009-10-29 19:43:48 +00:00
|
|
|
qDebug() << "[Utils] executable:" << executable;
|
2009-04-29 21:27:01 +00:00
|
|
|
if(QFileInfo(executable).isExecutable())
|
|
|
|
{
|
|
|
|
return QDir::toNativeSeparators(executable);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-08-10 19:46:51 +00:00
|
|
|
/** @brief checks different Enviroment things. Ask if user wants to continue.
|
|
|
|
* @param permission if it should check for permission
|
|
|
|
* @return string with error messages if problems occurred, empty strings if none.
|
|
|
|
*/
|
2010-04-02 21:24:19 +00:00
|
|
|
QString Utils::checkEnvironment(bool permission)
|
2009-08-10 19:46:51 +00:00
|
|
|
{
|
|
|
|
QString text = "";
|
|
|
|
|
|
|
|
// check permission
|
|
|
|
if(permission)
|
|
|
|
{
|
|
|
|
#if defined(Q_OS_WIN32)
|
2009-08-11 20:40:02 +00:00
|
|
|
if(System::userPermissions() != System::ADMIN)
|
2009-08-10 19:46:51 +00:00
|
|
|
{
|
2010-06-18 19:04:06 +00:00
|
|
|
text += tr("<li>Permissions insufficient for bootloader "
|
2009-08-10 19:46:51 +00:00
|
|
|
"installation.\nAdministrator priviledges are necessary.</li>");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check TargetId
|
|
|
|
RockboxInfo rbinfo(RbSettings::value(RbSettings::Mountpoint).toString());
|
|
|
|
QString installed = rbinfo.target();
|
2010-02-20 22:11:08 +00:00
|
|
|
if(!installed.isEmpty() && installed !=
|
|
|
|
SystemInfo::value(SystemInfo::CurConfigureModel).toString())
|
2009-08-10 19:46:51 +00:00
|
|
|
{
|
2010-06-18 19:04:06 +00:00
|
|
|
text += tr("<li>Target mismatch detected.\n"
|
2009-08-10 19:46:51 +00:00
|
|
|
"Installed target: %1, selected target: %2.</li>")
|
2010-01-24 21:12:11 +00:00
|
|
|
.arg(installed, SystemInfo::value(SystemInfo::CurPlatformName).toString());
|
2009-08-10 19:46:51 +00:00
|
|
|
// FIXME: replace installed by human-friendly name
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!text.isEmpty())
|
2010-06-18 19:04:06 +00:00
|
|
|
return tr("Problem detected:") + "<ul>" + text + "</ul>";
|
2009-08-10 19:46:51 +00:00
|
|
|
else
|
|
|
|
return text;
|
|
|
|
}
|
2010-06-11 19:18:13 +00:00
|
|
|
/** @brief Compare two version strings.
|
|
|
|
* @param s1 first version string
|
|
|
|
* @param s2 second version string
|
|
|
|
* @return 0 if strings identical, 1 if second is newer, -1 if first.
|
|
|
|
*/
|
|
|
|
int Utils::compareVersionStrings(QString s1, QString s2)
|
|
|
|
{
|
2010-06-30 21:59:47 +00:00
|
|
|
qDebug() << "[Utils] comparing version strings" << s1 << "and" << s2;
|
2010-06-11 19:18:13 +00:00
|
|
|
QString a = s1.trimmed();
|
|
|
|
QString b = s2.trimmed();
|
|
|
|
// if strings are identical return 0.
|
|
|
|
if(a.isEmpty())
|
|
|
|
return 1;
|
|
|
|
if(b.isEmpty())
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
while(!a.isEmpty() || !b.isEmpty()) {
|
|
|
|
// trim all leading non-digits and non-dots (dots are removed afterwards)
|
|
|
|
a.remove(QRegExp("^[^\\d\\.]*"));
|
|
|
|
b.remove(QRegExp("^[^\\d\\.]*"));
|
|
|
|
|
|
|
|
// trim all trailing non-digits for conversion (QString::toInt()
|
|
|
|
// requires this). Copy strings first as replace() changes the string.
|
|
|
|
QString numa = a;
|
|
|
|
QString numb = b;
|
|
|
|
numa.remove(QRegExp("\\D+.*$"));
|
|
|
|
numb.remove(QRegExp("\\D+.*$"));
|
|
|
|
|
|
|
|
// convert to number
|
|
|
|
bool ok1, ok2;
|
|
|
|
int vala = numa.toUInt(&ok1);
|
|
|
|
int valb = numb.toUInt(&ok2);
|
|
|
|
// if none of the numbers converted successfully we're at trailing garbage.
|
|
|
|
if(!ok1 && !ok2)
|
|
|
|
break;
|
|
|
|
if(!ok1)
|
|
|
|
return 1;
|
|
|
|
if(!ok2)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
// if numbers mismatch we have a decision.
|
|
|
|
if(vala != valb)
|
|
|
|
return (vala > valb) ? -1 : 1;
|
|
|
|
|
|
|
|
// trim leading digits.
|
|
|
|
a.remove(QRegExp("^\\d*"));
|
|
|
|
b.remove(QRegExp("^\\d*"));
|
|
|
|
|
2010-06-30 21:59:47 +00:00
|
|
|
// If only one of the following characters is a dot that one is
|
|
|
|
// "greater" then anything else. Make sure it's followed by a number,
|
|
|
|
// Otherwise it might be the end of the string or suffix. Do this
|
|
|
|
// before version addon characters check to avoid stopping too early.
|
|
|
|
bool adot = a.contains(QRegExp("^[a-zA-Z]*\\.[0-9]"));
|
|
|
|
bool bdot = b.contains(QRegExp("^[a-zA-Z]*\\.[0-9]"));
|
|
|
|
if(adot && !bdot)
|
|
|
|
return -1;
|
|
|
|
if(!adot && bdot)
|
|
|
|
return 1;
|
2010-06-11 19:18:13 +00:00
|
|
|
// if number is immediately followed by a character consider it as
|
2010-06-30 21:59:47 +00:00
|
|
|
// version addon (like 1.2.3b). In this case compare characters and end
|
|
|
|
// (version numbers like 1.2b.3 aren't handled).
|
2010-06-11 19:18:13 +00:00
|
|
|
QChar ltra;
|
|
|
|
QChar ltrb;
|
|
|
|
if(a.contains(QRegExp("^[a-zA-Z]")))
|
|
|
|
ltra = a.at(0);
|
|
|
|
if(b.contains(QRegExp("^[a-zA-Z]")))
|
|
|
|
ltrb = b.at(0);
|
|
|
|
if(ltra != ltrb)
|
|
|
|
return (ltra < ltrb) ? 1 : -1;
|
2010-06-30 21:59:47 +00:00
|
|
|
|
2010-06-11 19:18:13 +00:00
|
|
|
// both are identical or no addon characters, ignore.
|
|
|
|
// remove modifiers and following dot.
|
|
|
|
a.remove(QRegExp("^[a-zA-Z]*\\."));
|
|
|
|
b.remove(QRegExp("^[a-zA-Z]*\\."));
|
|
|
|
}
|
|
|
|
|
|
|
|
// no differences found.
|
|
|
|
return 0;
|
|
|
|
}
|
2009-08-10 19:46:51 +00:00
|
|
|
|