|
1 /**************************************************************************** |
|
2 ** |
|
3 ** Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies). |
|
4 ** All rights reserved. |
|
5 ** Contact: Nokia Corporation (qt-info@nokia.com) |
|
6 ** |
|
7 ** This file is part of the tools applications of the Qt Toolkit. |
|
8 ** |
|
9 ** $QT_BEGIN_LICENSE:LGPL$ |
|
10 ** No Commercial Usage |
|
11 ** This file contains pre-release code and may not be distributed. |
|
12 ** You may use this file in accordance with the terms and conditions |
|
13 ** contained in the Technology Preview License Agreement accompanying |
|
14 ** this package. |
|
15 ** |
|
16 ** GNU Lesser General Public License Usage |
|
17 ** Alternatively, this file may be used under the terms of the GNU Lesser |
|
18 ** General Public License version 2.1 as published by the Free Software |
|
19 ** Foundation and appearing in the file LICENSE.LGPL included in the |
|
20 ** packaging of this file. Please review the following information to |
|
21 ** ensure the GNU Lesser General Public License version 2.1 requirements |
|
22 ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. |
|
23 ** |
|
24 ** In addition, as a special exception, Nokia gives you certain additional |
|
25 ** rights. These rights are described in the Nokia Qt LGPL Exception |
|
26 ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. |
|
27 ** |
|
28 ** If you have questions regarding the use of this file, please contact |
|
29 ** Nokia at qt-info@nokia.com. |
|
30 ** |
|
31 ** |
|
32 ** |
|
33 ** |
|
34 ** |
|
35 ** |
|
36 ** |
|
37 ** |
|
38 ** $QT_END_LICENSE$ |
|
39 ** |
|
40 ****************************************************************************/ |
|
41 |
|
42 #include "environment.h" |
|
43 |
|
44 #include <process.h> |
|
45 #include <iostream> |
|
46 #include <qdebug.h> |
|
47 #include <QDir> |
|
48 #include <QStringList> |
|
49 #include <QMap> |
|
50 #include <QDir> |
|
51 #include <QFile> |
|
52 #include <QFileInfo> |
|
53 |
|
54 //#define CONFIGURE_DEBUG_EXECUTE |
|
55 //#define CONFIGURE_DEBUG_CP_DIR |
|
56 |
|
57 using namespace std; |
|
58 |
|
59 #ifdef Q_OS_WIN32 |
|
60 #include <qt_windows.h> |
|
61 #endif |
|
62 |
|
63 |
|
64 QT_BEGIN_NAMESPACE |
|
65 |
|
66 struct CompilerInfo{ |
|
67 Compiler compiler; |
|
68 const char *compilerStr; |
|
69 const char *regKey; |
|
70 const char *executable; |
|
71 } compiler_info[] = { |
|
72 // The compilers here are sorted in a reversed-preferred order |
|
73 {CC_BORLAND, "Borland C++", 0, "bcc32.exe"}, |
|
74 {CC_MINGW, "MinGW (Minimalist GNU for Windows)", 0, "mingw32-gcc.exe"}, |
|
75 {CC_INTEL, "Intel(R) C++ Compiler for 32-bit applications", 0, "icl.exe"}, // xilink.exe, xilink5.exe, xilink6.exe, xilib.exe |
|
76 {CC_MSVC6, "Microsoft (R) 32-bit C/C++ Optimizing Compiler (6.x)", "Software\\Microsoft\\VisualStudio\\6.0\\Setup\\Microsoft Visual C++\\ProductDir", "cl.exe"}, // link.exe, lib.exe |
|
77 {CC_NET2002, "Microsoft (R) 32-bit C/C++ Optimizing Compiler.NET 2002 (7.0)", "Software\\Microsoft\\VisualStudio\\7.0\\Setup\\VC\\ProductDir", "cl.exe"}, // link.exe, lib.exe |
|
78 {CC_NET2003, "Microsoft (R) 32-bit C/C++ Optimizing Compiler.NET 2003 (7.1)", "Software\\Microsoft\\VisualStudio\\7.1\\Setup\\VC\\ProductDir", "cl.exe"}, // link.exe, lib.exe |
|
79 {CC_NET2005, "Microsoft (R) 32-bit C/C++ Optimizing Compiler.NET 2005 (8.0)", "Software\\Microsoft\\VisualStudio\\SxS\\VC7\\8.0", "cl.exe"}, // link.exe, lib.exe |
|
80 {CC_NET2008, "Microsoft (R) 32-bit C/C++ Optimizing Compiler.NET 2008 (9.0)", "Software\\Microsoft\\VisualStudio\\SxS\\VC7\\9.0", "cl.exe"}, // link.exe, lib.exe |
|
81 {CC_UNKNOWN, "Unknown", 0, 0}, |
|
82 }; |
|
83 |
|
84 |
|
85 // Initialize static variables |
|
86 Compiler Environment::detectedCompiler = CC_UNKNOWN; |
|
87 |
|
88 /*! |
|
89 Returns the pointer to the CompilerInfo for a \a compiler. |
|
90 */ |
|
91 CompilerInfo *Environment::compilerInfo(Compiler compiler) |
|
92 { |
|
93 int i = 0; |
|
94 while(compiler_info[i].compiler != compiler && compiler_info[i].compiler != CC_UNKNOWN) |
|
95 ++i; |
|
96 return &(compiler_info[i]); |
|
97 } |
|
98 |
|
99 /*! |
|
100 Returns the path part of a registry key. |
|
101 Ei. |
|
102 For a key |
|
103 "Software\\Microsoft\\VisualStudio\\8.0\\Setup\\VC\\ProductDir" |
|
104 it returns |
|
105 "Software\\Microsoft\\VisualStudio\\8.0\\Setup\\VC\\" |
|
106 */ |
|
107 QString Environment::keyPath(const QString &rKey) |
|
108 { |
|
109 int idx = rKey.lastIndexOf(QLatin1Char('\\')); |
|
110 if (idx == -1) |
|
111 return QString(); |
|
112 return rKey.left(idx + 1); |
|
113 } |
|
114 |
|
115 /*! |
|
116 Returns the name part of a registry key. |
|
117 Ei. |
|
118 For a key |
|
119 "Software\\Microsoft\\VisualStudio\\8.0\\Setup\\VC\\ProductDir" |
|
120 it returns |
|
121 "ProductDir" |
|
122 */ |
|
123 QString Environment::keyName(const QString &rKey) |
|
124 { |
|
125 int idx = rKey.lastIndexOf(QLatin1Char('\\')); |
|
126 if (idx == -1) |
|
127 return rKey; |
|
128 |
|
129 QString res(rKey.mid(idx + 1)); |
|
130 if (res == "Default" || res == ".") |
|
131 res = ""; |
|
132 return res; |
|
133 } |
|
134 |
|
135 /*! |
|
136 Returns a registry keys value in string form. |
|
137 If the registry key does not exist, or cannot be accessed, a |
|
138 QString() is returned. |
|
139 */ |
|
140 QString Environment::readRegistryKey(HKEY parentHandle, const QString &rSubkey) |
|
141 { |
|
142 #ifndef Q_OS_WIN32 |
|
143 return QString(); |
|
144 #else |
|
145 QString rSubkeyName = keyName(rSubkey); |
|
146 QString rSubkeyPath = keyPath(rSubkey); |
|
147 |
|
148 HKEY handle = 0; |
|
149 LONG res = RegOpenKeyEx(parentHandle, (wchar_t*)rSubkeyPath.utf16(), 0, KEY_READ, &handle); |
|
150 if (res != ERROR_SUCCESS) |
|
151 return QString(); |
|
152 |
|
153 // get the size and type of the value |
|
154 DWORD dataType; |
|
155 DWORD dataSize; |
|
156 res = RegQueryValueEx(handle, (wchar_t*)rSubkeyName.utf16(), 0, &dataType, 0, &dataSize); |
|
157 if (res != ERROR_SUCCESS) { |
|
158 RegCloseKey(handle); |
|
159 return QString(); |
|
160 } |
|
161 |
|
162 // get the value |
|
163 QByteArray data(dataSize, 0); |
|
164 res = RegQueryValueEx(handle, (wchar_t*)rSubkeyName.utf16(), 0, 0, |
|
165 reinterpret_cast<unsigned char*>(data.data()), &dataSize); |
|
166 if (res != ERROR_SUCCESS) { |
|
167 RegCloseKey(handle); |
|
168 return QString(); |
|
169 } |
|
170 |
|
171 QString result; |
|
172 switch (dataType) { |
|
173 case REG_EXPAND_SZ: |
|
174 case REG_SZ: { |
|
175 result = QString::fromWCharArray(((const wchar_t *)data.constData())); |
|
176 break; |
|
177 } |
|
178 |
|
179 case REG_MULTI_SZ: { |
|
180 QStringList l; |
|
181 int i = 0; |
|
182 for (;;) { |
|
183 QString s = QString::fromWCharArray((const wchar_t *)data.constData() + i); |
|
184 i += s.length() + 1; |
|
185 |
|
186 if (s.isEmpty()) |
|
187 break; |
|
188 l.append(s); |
|
189 } |
|
190 result = l.join(", "); |
|
191 break; |
|
192 } |
|
193 |
|
194 case REG_NONE: |
|
195 case REG_BINARY: { |
|
196 result = QString::fromWCharArray((const wchar_t *)data.constData(), data.size() / 2); |
|
197 break; |
|
198 } |
|
199 |
|
200 case REG_DWORD_BIG_ENDIAN: |
|
201 case REG_DWORD: { |
|
202 Q_ASSERT(data.size() == sizeof(int)); |
|
203 int i; |
|
204 memcpy((char*)&i, data.constData(), sizeof(int)); |
|
205 result = QString::number(i); |
|
206 break; |
|
207 } |
|
208 |
|
209 default: |
|
210 qWarning("QSettings: unknown data %d type in windows registry", dataType); |
|
211 break; |
|
212 } |
|
213 |
|
214 RegCloseKey(handle); |
|
215 return result; |
|
216 #endif |
|
217 } |
|
218 |
|
219 /*! |
|
220 Returns the qmakespec for the compiler detected on the system. |
|
221 */ |
|
222 QString Environment::detectQMakeSpec() |
|
223 { |
|
224 QString spec; |
|
225 switch (detectCompiler()) { |
|
226 case CC_NET2008: |
|
227 spec = "win32-msvc2008"; |
|
228 break; |
|
229 case CC_NET2005: |
|
230 spec = "win32-msvc2005"; |
|
231 break; |
|
232 case CC_NET2003: |
|
233 spec = "win32-msvc2003"; |
|
234 break; |
|
235 case CC_NET2002: |
|
236 spec = "win32-msvc2002"; |
|
237 break; |
|
238 case CC_MSVC4: |
|
239 case CC_MSVC5: |
|
240 case CC_MSVC6: |
|
241 spec = "win32-msvc"; |
|
242 break; |
|
243 case CC_INTEL: |
|
244 spec = "win32-icc"; |
|
245 break; |
|
246 case CC_MINGW: |
|
247 spec = "win32-g++"; |
|
248 break; |
|
249 case CC_BORLAND: |
|
250 spec = "win32-borland"; |
|
251 break; |
|
252 default: |
|
253 break; |
|
254 } |
|
255 |
|
256 return spec; |
|
257 } |
|
258 |
|
259 /*! |
|
260 Returns the enum of the compiler which was detected on the system. |
|
261 The compilers are detected in the order as entered into the |
|
262 compiler_info list. |
|
263 |
|
264 If more than one compiler is found, CC_UNKNOWN is returned. |
|
265 */ |
|
266 Compiler Environment::detectCompiler() |
|
267 { |
|
268 #ifndef Q_OS_WIN32 |
|
269 return MSVC6; // Always generate MSVC 6.0 versions on other platforms |
|
270 #else |
|
271 if(detectedCompiler != CC_UNKNOWN) |
|
272 return detectedCompiler; |
|
273 |
|
274 int installed = 0; |
|
275 |
|
276 // Check for compilers in registry first, to see which version is in PATH |
|
277 QString paths = qgetenv("PATH"); |
|
278 QStringList pathlist = paths.toLower().split(";"); |
|
279 for(int i = 0; compiler_info[i].compiler; ++i) { |
|
280 QString productPath = readRegistryKey(HKEY_LOCAL_MACHINE, compiler_info[i].regKey).toLower(); |
|
281 if (productPath.length()) { |
|
282 QStringList::iterator it; |
|
283 for(it = pathlist.begin(); it != pathlist.end(); ++it) { |
|
284 if((*it).contains(productPath)) { |
|
285 ++installed; |
|
286 detectedCompiler = compiler_info[i].compiler; |
|
287 break; |
|
288 } |
|
289 } |
|
290 } |
|
291 } |
|
292 |
|
293 // Now just go looking for the executables, and accept any executable as the lowest version |
|
294 if (!installed) { |
|
295 for(int i = 0; compiler_info[i].compiler; ++i) { |
|
296 QString executable = QString(compiler_info[i].executable).toLower(); |
|
297 if (executable.length() && Environment::detectExecutable(executable)) { |
|
298 ++installed; |
|
299 detectedCompiler = compiler_info[i].compiler; |
|
300 break; |
|
301 } |
|
302 } |
|
303 } |
|
304 |
|
305 if (installed > 1) { |
|
306 cout << "Found more than one known compiler! Using \"" << compilerInfo(detectedCompiler)->compilerStr << "\"" << endl; |
|
307 detectedCompiler = CC_UNKNOWN; |
|
308 } |
|
309 return detectedCompiler; |
|
310 #endif |
|
311 }; |
|
312 |
|
313 /*! |
|
314 Returns true if the \a executable could be loaded, else false. |
|
315 This means that the executable either is in the current directory |
|
316 or in the PATH. |
|
317 */ |
|
318 bool Environment::detectExecutable(const QString &executable) |
|
319 { |
|
320 PROCESS_INFORMATION procInfo; |
|
321 memset(&procInfo, 0, sizeof(procInfo)); |
|
322 |
|
323 STARTUPINFO startInfo; |
|
324 memset(&startInfo, 0, sizeof(startInfo)); |
|
325 startInfo.cb = sizeof(startInfo); |
|
326 |
|
327 bool couldExecute = CreateProcess(0, (wchar_t*)executable.utf16(), |
|
328 0, 0, false, |
|
329 CREATE_NO_WINDOW | CREATE_SUSPENDED, |
|
330 0, 0, &startInfo, &procInfo); |
|
331 |
|
332 if (couldExecute) { |
|
333 CloseHandle(procInfo.hThread); |
|
334 TerminateProcess(procInfo.hProcess, 0); |
|
335 CloseHandle(procInfo.hProcess); |
|
336 } |
|
337 return couldExecute; |
|
338 } |
|
339 |
|
340 /*! |
|
341 Creates a commandling from \a program and it \a arguments, |
|
342 escaping characters that needs it. |
|
343 */ |
|
344 static QString qt_create_commandline(const QString &program, const QStringList &arguments) |
|
345 { |
|
346 QString programName = program; |
|
347 if (!programName.startsWith("\"") && !programName.endsWith("\"") && programName.contains(" ")) |
|
348 programName = "\"" + programName + "\""; |
|
349 programName.replace("/", "\\"); |
|
350 |
|
351 QString args; |
|
352 // add the prgram as the first arrg ... it works better |
|
353 args = programName + " "; |
|
354 for (int i=0; i<arguments.size(); ++i) { |
|
355 QString tmp = arguments.at(i); |
|
356 // in the case of \" already being in the string the \ must also be escaped |
|
357 tmp.replace( "\\\"", "\\\\\"" ); |
|
358 // escape a single " because the arguments will be parsed |
|
359 tmp.replace( "\"", "\\\"" ); |
|
360 if (tmp.isEmpty() || tmp.contains(' ') || tmp.contains('\t')) { |
|
361 // The argument must not end with a \ since this would be interpreted |
|
362 // as escaping the quote -- rather put the \ behind the quote: e.g. |
|
363 // rather use "foo"\ than "foo\" |
|
364 QString endQuote("\""); |
|
365 int i = tmp.length(); |
|
366 while (i>0 && tmp.at(i-1) == '\\') { |
|
367 --i; |
|
368 endQuote += "\\"; |
|
369 } |
|
370 args += QString(" \"") + tmp.left(i) + endQuote; |
|
371 } else { |
|
372 args += ' ' + tmp; |
|
373 } |
|
374 } |
|
375 return args; |
|
376 } |
|
377 |
|
378 /*! |
|
379 Creates a QByteArray of the \a environment. |
|
380 */ |
|
381 static QByteArray qt_create_environment(const QStringList &environment) |
|
382 { |
|
383 QByteArray envlist; |
|
384 if (environment.isEmpty()) |
|
385 return envlist; |
|
386 |
|
387 int pos = 0; |
|
388 // add PATH if necessary (for DLL loading) |
|
389 QByteArray path = qgetenv("PATH"); |
|
390 if (environment.filter(QRegExp("^PATH=",Qt::CaseInsensitive)).isEmpty() && !path.isNull()) { |
|
391 QString tmp = QString(QLatin1String("PATH=%1")).arg(QString::fromLocal8Bit(path)); |
|
392 uint tmpSize = sizeof(wchar_t) * (tmp.length() + 1); |
|
393 envlist.resize(envlist.size() + tmpSize); |
|
394 memcpy(envlist.data() + pos, tmp.utf16(), tmpSize); |
|
395 pos += tmpSize; |
|
396 } |
|
397 // add the user environment |
|
398 for (QStringList::ConstIterator it = environment.begin(); it != environment.end(); it++ ) { |
|
399 QString tmp = *it; |
|
400 uint tmpSize = sizeof(wchar_t) * (tmp.length() + 1); |
|
401 envlist.resize(envlist.size() + tmpSize); |
|
402 memcpy(envlist.data() + pos, tmp.utf16(), tmpSize); |
|
403 pos += tmpSize; |
|
404 } |
|
405 // add the 2 terminating 0 (actually 4, just to be on the safe side) |
|
406 envlist.resize(envlist.size() + 4); |
|
407 envlist[pos++] = 0; |
|
408 envlist[pos++] = 0; |
|
409 envlist[pos++] = 0; |
|
410 envlist[pos++] = 0; |
|
411 |
|
412 return envlist; |
|
413 } |
|
414 |
|
415 /*! |
|
416 Executes the command described in \a arguments, in the |
|
417 environment inherited from the parent process, with the |
|
418 \a additionalEnv settings applied. |
|
419 \a removeEnv removes the specified environment variables from |
|
420 the environment of the executed process. |
|
421 |
|
422 Returns the exit value of the process, or -1 if the command could |
|
423 not be executed. |
|
424 |
|
425 This function uses _(w)spawnvpe to spawn a process by searching |
|
426 through the PATH environment variable. |
|
427 */ |
|
428 int Environment::execute(QStringList arguments, const QStringList &additionalEnv, const QStringList &removeEnv) |
|
429 { |
|
430 #ifdef CONFIGURE_DEBUG_EXECUTE |
|
431 qDebug() << "About to Execute: " << arguments; |
|
432 qDebug() << " " << QDir::currentPath(); |
|
433 qDebug() << " " << additionalEnv; |
|
434 qDebug() << " " << removeEnv; |
|
435 #endif |
|
436 // Create the full environment from the current environment and |
|
437 // the additionalEnv strings, then remove all variables defined |
|
438 // in removeEnv |
|
439 QMap<QString, QString> fullEnvMap; |
|
440 LPWSTR envStrings = GetEnvironmentStrings(); |
|
441 if (envStrings) { |
|
442 int strLen = 0; |
|
443 for (LPWSTR envString = envStrings; *(envString); envString += strLen + 1) { |
|
444 strLen = wcslen(envString); |
|
445 QString str = QString((const QChar*)envString, strLen); |
|
446 if (!str.startsWith("=")) { // These are added by the system |
|
447 int sepIndex = str.indexOf('='); |
|
448 fullEnvMap.insert(str.left(sepIndex).toUpper(), str.mid(sepIndex +1)); |
|
449 } |
|
450 } |
|
451 } |
|
452 FreeEnvironmentStrings(envStrings); |
|
453 |
|
454 // Add additionalEnv variables |
|
455 for (int i = 0; i < additionalEnv.count(); ++i) { |
|
456 const QString &str = additionalEnv.at(i); |
|
457 int sepIndex = str.indexOf('='); |
|
458 fullEnvMap.insert(str.left(sepIndex).toUpper(), str.mid(sepIndex +1)); |
|
459 } |
|
460 |
|
461 // Remove removeEnv variables |
|
462 for (int j = 0; j < removeEnv.count(); ++j) |
|
463 fullEnvMap.remove(removeEnv.at(j).toUpper()); |
|
464 |
|
465 // Add all variables to a QStringList |
|
466 QStringList fullEnv; |
|
467 QMapIterator<QString, QString> it(fullEnvMap); |
|
468 while (it.hasNext()) { |
|
469 it.next(); |
|
470 fullEnv += QString(it.key() + "=" + it.value()); |
|
471 } |
|
472 |
|
473 // ---------------------------- |
|
474 QString program = arguments.takeAt(0); |
|
475 QString args = qt_create_commandline(program, arguments); |
|
476 QByteArray envlist = qt_create_environment(fullEnv); |
|
477 |
|
478 DWORD exitCode = -1; |
|
479 PROCESS_INFORMATION procInfo; |
|
480 memset(&procInfo, 0, sizeof(procInfo)); |
|
481 |
|
482 STARTUPINFO startInfo; |
|
483 memset(&startInfo, 0, sizeof(startInfo)); |
|
484 startInfo.cb = sizeof(startInfo); |
|
485 |
|
486 bool couldExecute = CreateProcess(0, (wchar_t*)args.utf16(), |
|
487 0, 0, true, CREATE_UNICODE_ENVIRONMENT, |
|
488 envlist.isEmpty() ? 0 : envlist.data(), |
|
489 0, &startInfo, &procInfo); |
|
490 |
|
491 if (couldExecute) { |
|
492 WaitForSingleObject(procInfo.hProcess, INFINITE); |
|
493 GetExitCodeProcess(procInfo.hProcess, &exitCode); |
|
494 CloseHandle(procInfo.hThread); |
|
495 CloseHandle(procInfo.hProcess); |
|
496 } |
|
497 |
|
498 |
|
499 if (exitCode == -1) { |
|
500 switch(GetLastError()) { |
|
501 case E2BIG: |
|
502 cerr << "execute: Argument list exceeds 1024 bytes" << endl; |
|
503 foreach(QString arg, arguments) |
|
504 cerr << " (" << arg.toLocal8Bit().constData() << ")" << endl; |
|
505 break; |
|
506 case ENOENT: |
|
507 cerr << "execute: File or path is not found (" << program.toLocal8Bit().constData() << ")" << endl; |
|
508 break; |
|
509 case ENOEXEC: |
|
510 cerr << "execute: Specified file is not executable or has invalid executable-file format (" << program.toLocal8Bit().constData() << ")" << endl; |
|
511 break; |
|
512 case ENOMEM: |
|
513 cerr << "execute: Not enough memory is available to execute new process." << endl; |
|
514 break; |
|
515 default: |
|
516 cerr << "execute: Unknown error" << endl; |
|
517 foreach(QString arg, arguments) |
|
518 cerr << " (" << arg.toLocal8Bit().constData() << ")" << endl; |
|
519 break; |
|
520 } |
|
521 } |
|
522 return exitCode; |
|
523 } |
|
524 |
|
525 bool Environment::cpdir(const QString &srcDir, const QString &destDir) |
|
526 { |
|
527 QString cleanSrcName = QDir::cleanPath(srcDir); |
|
528 QString cleanDstName = QDir::cleanPath(destDir); |
|
529 #ifdef CONFIGURE_DEBUG_CP_DIR |
|
530 qDebug() << "Attempt to cpdir " << cleanSrcName << "->" << cleanDstName; |
|
531 #endif |
|
532 if(!QFile::exists(cleanDstName) && !QDir().mkpath(cleanDstName)) { |
|
533 qDebug() << "cpdir: Failure to create " << cleanDstName; |
|
534 return false; |
|
535 } |
|
536 |
|
537 bool result = true; |
|
538 QDir dir = QDir(cleanSrcName); |
|
539 QFileInfoList allEntries = dir.entryInfoList(QDir::AllDirs | QDir::Files | QDir::NoDotAndDotDot); |
|
540 for (int i = 0; result && (i < allEntries.count()); ++i) { |
|
541 QFileInfo entry = allEntries.at(i); |
|
542 bool intermediate = true; |
|
543 if (entry.isDir()) { |
|
544 intermediate = cpdir(QString("%1/%2").arg(cleanSrcName).arg(entry.fileName()), |
|
545 QString("%1/%2").arg(cleanDstName).arg(entry.fileName())); |
|
546 } else { |
|
547 QString destFile = QString("%1/%2").arg(cleanDstName).arg(entry.fileName()); |
|
548 #ifdef CONFIGURE_DEBUG_CP_DIR |
|
549 qDebug() << "About to cp (file)" << entry.absoluteFilePath() << "->" << destFile; |
|
550 #endif |
|
551 QFile::remove(destFile); |
|
552 intermediate = QFile::copy(entry.absoluteFilePath(), destFile); |
|
553 SetFileAttributes((wchar_t*)destFile.utf16(), FILE_ATTRIBUTE_NORMAL); |
|
554 } |
|
555 if(!intermediate) { |
|
556 qDebug() << "cpdir: Failure for " << entry.fileName() << entry.isDir(); |
|
557 result = false; |
|
558 } |
|
559 } |
|
560 return result; |
|
561 } |
|
562 |
|
563 bool Environment::rmdir(const QString &name) |
|
564 { |
|
565 bool result = true; |
|
566 QString cleanName = QDir::cleanPath(name); |
|
567 |
|
568 QDir dir = QDir(cleanName); |
|
569 QFileInfoList allEntries = dir.entryInfoList(QDir::AllDirs | QDir::Files | QDir::NoDotAndDotDot); |
|
570 for (int i = 0; result && (i < allEntries.count()); ++i) { |
|
571 QFileInfo entry = allEntries.at(i); |
|
572 if (entry.isDir()) { |
|
573 result &= rmdir(entry.absoluteFilePath()); |
|
574 } else { |
|
575 result &= QFile::remove(entry.absoluteFilePath()); |
|
576 } |
|
577 } |
|
578 result &= dir.rmdir(cleanName); |
|
579 return result; |
|
580 } |
|
581 |
|
582 QT_END_NAMESPACE |