source: trunk/qmake/generators/unix/unixmake.cpp

Last change on this file was 846, checked in by Dmitry A. Kuminov, 14 years ago

trunk: Merged in qt 4.7.2 sources from branches/vendor/nokia/qt.

File size: 42.3 KB
Line 
1/****************************************************************************
2**
3** Copyright (C) 2011 Nokia Corporation and/or its subsidiary(-ies).
4** All rights reserved.
5** Contact: Nokia Corporation ([email protected])
6**
7** This file is part of the qmake application of the Qt Toolkit.
8**
9** $QT_BEGIN_LICENSE:LGPL$
10** Commercial Usage
11** Licensees holding valid Qt Commercial licenses may use this file in
12** accordance with the Qt Commercial License Agreement provided with the
13** Software or, alternatively, in accordance with the terms contained in
14** a written agreement between you and Nokia.
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** GNU General Public License Usage
29** Alternatively, this file may be used under the terms of the GNU
30** General Public License version 3.0 as published by the Free Software
31** Foundation and appearing in the file LICENSE.GPL included in the
32** packaging of this file. Please review the following information to
33** ensure the GNU General Public License version 3.0 requirements will be
34** met: http://www.gnu.org/copyleft/gpl.html.
35**
36** If you have questions regarding the use of this file, please contact
37** Nokia at [email protected].
38** $QT_END_LICENSE$
39**
40****************************************************************************/
41
42#include "unixmake.h"
43#include "option.h"
44#include <qregexp.h>
45#include <qfile.h>
46#include <qhash.h>
47#include <qdir.h>
48#include <time.h>
49#include <qdebug.h>
50
51QT_BEGIN_NAMESPACE
52
53void
54UnixMakefileGenerator::init()
55{
56 if(init_flag)
57 return;
58 init_flag = true;
59
60 if(project->isEmpty("QMAKE_EXTENSION_SHLIB")) {
61 if(project->isEmpty("QMAKE_CYGWIN_SHLIB")) {
62 project->values("QMAKE_EXTENSION_SHLIB").append("so");
63 } else {
64 project->values("QMAKE_EXTENSION_SHLIB").append("dll");
65 }
66 }
67
68 if (project->isEmpty("QMAKE_PREFIX_SHLIB"))
69 // Prevent crash when using the empty variable.
70 project->values("QMAKE_PREFIX_SHLIB").append("");
71
72 if(!project->isEmpty("QMAKE_FAILED_REQUIREMENTS")) /* no point */
73 return;
74
75 QStringList &configs = project->values("CONFIG");
76 if(project->isEmpty("ICON") && !project->isEmpty("RC_FILE"))
77 project->values("ICON") = project->values("RC_FILE");
78 if(project->isEmpty("QMAKE_EXTENSION_PLUGIN"))
79 project->values("QMAKE_EXTENSION_PLUGIN").append(project->first("QMAKE_EXTENSION_SHLIB"));
80 if(project->isEmpty("QMAKE_COPY_FILE"))
81 project->values("QMAKE_COPY_FILE").append("$(COPY)");
82 if(project->isEmpty("QMAKE_STREAM_EDITOR"))
83 project->values("QMAKE_STREAM_EDITOR").append("sed");
84 if(project->isEmpty("QMAKE_COPY_DIR"))
85 project->values("QMAKE_COPY_DIR").append("$(COPY) -R");
86 if(project->isEmpty("QMAKE_INSTALL_FILE"))
87 project->values("QMAKE_INSTALL_FILE").append("$(COPY_FILE)");
88 if(project->isEmpty("QMAKE_INSTALL_DIR"))
89 project->values("QMAKE_INSTALL_DIR").append("$(COPY_DIR)");
90 if(project->isEmpty("QMAKE_INSTALL_PROGRAM"))
91 project->values("QMAKE_INSTALL_PROGRAM").append("$(COPY_FILE)");
92 if(project->isEmpty("QMAKE_LIBTOOL"))
93 project->values("QMAKE_LIBTOOL").append("libtool --silent");
94 if(project->isEmpty("QMAKE_SYMBOLIC_LINK"))
95 project->values("QMAKE_SYMBOLIC_LINK").append("ln -f -s");
96
97 /* this should probably not be here, but I'm using it to wrap the .t files */
98 if(project->first("TEMPLATE") == "app")
99 project->values("QMAKE_APP_FLAG").append("1");
100 else if(project->first("TEMPLATE") == "lib")
101 project->values("QMAKE_LIB_FLAG").append("1");
102 else if(project->first("TEMPLATE") == "subdirs") {
103 MakefileGenerator::init();
104 if(project->isEmpty("MAKEFILE"))
105 project->values("MAKEFILE").append("Makefile");
106 if(project->values("QMAKE_INTERNAL_QMAKE_DEPS").indexOf("qmake_all") == -1)
107 project->values("QMAKE_INTERNAL_QMAKE_DEPS").append("qmake_all");
108 return; /* subdirs is done */
109 }
110
111 //If the TARGET looks like a path split it into DESTDIR and the resulting TARGET
112 if(!project->isEmpty("TARGET")) {
113 project->values("TARGET") = escapeFilePaths(project->values("TARGET"));
114 QString targ = unescapeFilePath(project->first("TARGET"));
115 int slsh = qMax(targ.lastIndexOf('/'), targ.lastIndexOf(Option::dir_sep));
116 if(slsh != -1) {
117 if(project->isEmpty("DESTDIR"))
118 project->values("DESTDIR").append("");
119 else if(project->first("DESTDIR").right(1) != Option::dir_sep)
120 project->values("DESTDIR") = QStringList(project->first("DESTDIR") + Option::dir_sep);
121 project->values("DESTDIR") = QStringList(project->first("DESTDIR") + targ.left(slsh+1));
122 project->values("TARGET") = QStringList(targ.mid(slsh+1));
123 }
124 }
125
126 project->values("QMAKE_ORIG_TARGET") = project->values("TARGET");
127 project->values("QMAKE_ORIG_DESTDIR") = project->values("DESTDIR");
128 project->values("QMAKE_LIBS") += escapeFilePaths(project->values("LIBS"));
129 project->values("QMAKE_LIBS_PRIVATE") += escapeFilePaths(project->values("LIBS_PRIVATE"));
130 if((!project->isEmpty("QMAKE_LIB_FLAG") && !project->isActiveConfig("staticlib")) ||
131 (project->isActiveConfig("qt") && project->isActiveConfig("plugin"))) {
132 if(configs.indexOf("dll") == -1) configs.append("dll");
133 } else if(!project->isEmpty("QMAKE_APP_FLAG") || project->isActiveConfig("dll")) {
134 configs.removeAll("staticlib");
135 }
136 if(!project->isEmpty("QMAKE_INCREMENTAL"))
137 project->values("QMAKE_LFLAGS") += project->values("QMAKE_LFLAGS_INCREMENTAL");
138 else if(!project->isEmpty("QMAKE_LFLAGS_PREBIND") &&
139 !project->values("QMAKE_LIB_FLAG").isEmpty() &&
140 project->isActiveConfig("dll"))
141 project->values("QMAKE_LFLAGS") += project->values("QMAKE_LFLAGS_PREBIND");
142 if(!project->isEmpty("QMAKE_INCDIR"))
143 project->values("INCLUDEPATH") += project->values("QMAKE_INCDIR");
144 if(!project->isEmpty("QMAKE_LIBDIR")) {
145 const QStringList &libdirs = project->values("QMAKE_LIBDIR");
146 for(int i = 0; i < libdirs.size(); ++i) {
147 if(!project->isEmpty("QMAKE_LFLAGS_RPATH") && project->isActiveConfig("rpath_libdirs"))
148 project->values("QMAKE_LFLAGS") += var("QMAKE_LFLAGS_RPATH") + libdirs[i];
149 if (project->isActiveConfig("rvct_linker")) {
150 project->values("QMAKE_LIBDIR_FLAGS") += "--userlibpath " + escapeFilePath(libdirs[i]);
151 } else {
152 project->values("QMAKE_LIBDIR_FLAGS") += "-L" + escapeFilePath(libdirs[i]);
153 }
154 }
155 }
156 if(project->isActiveConfig("macx") && !project->isEmpty("QMAKE_FRAMEWORKPATH")) {
157 const QStringList &fwdirs = project->values("QMAKE_FRAMEWORKPATH");
158 for(int i = 0; i < fwdirs.size(); ++i) {
159 project->values("QMAKE_FRAMEWORKPATH_FLAGS") += "-F" + escapeFilePath(fwdirs[i]);
160 }
161 }
162 if(!project->isEmpty("QMAKE_RPATHDIR")) {
163 const QStringList &rpathdirs = project->values("QMAKE_RPATHDIR");
164 for(int i = 0; i < rpathdirs.size(); ++i) {
165 if(!project->isEmpty("QMAKE_LFLAGS_RPATH"))
166 project->values("QMAKE_LFLAGS") += var("QMAKE_LFLAGS_RPATH") + escapeFilePath(QFileInfo(rpathdirs[i]).absoluteFilePath());
167 }
168 }
169
170 project->values("QMAKE_FILETAGS") << "SOURCES" << "GENERATED_SOURCES" << "TARGET" << "DESTDIR";
171 if(!project->isEmpty("QMAKE_EXTRA_COMPILERS")) {
172 const QStringList &quc = project->values("QMAKE_EXTRA_COMPILERS");
173 for(int i = 0; i < quc.size(); ++i)
174 project->values("QMAKE_FILETAGS") += project->values(quc[i]+".input");
175 }
176
177 if(project->isActiveConfig("GNUmake") && !project->isEmpty("QMAKE_CFLAGS_DEPS"))
178 include_deps = true; //do not generate deps
179 if(project->isActiveConfig("compile_libtool"))
180 Option::obj_ext = ".lo"; //override the .o
181
182 MakefileGenerator::init();
183
184 QString comps[] = { "C", "CXX", "OBJC", "OBJCXX", QString() };
185 for(int i = 0; !comps[i].isNull(); i++) {
186 QString compile_flag = var("QMAKE_COMPILE_FLAG");
187 if(compile_flag.isEmpty())
188 compile_flag = "-c";
189
190 if(doPrecompiledHeaders() && !project->isEmpty("PRECOMPILED_HEADER")) {
191 QString pchFlags = var("QMAKE_" + comps[i] + "FLAGS_USE_PRECOMPILE");
192
193 QString pchBaseName;
194 if(!project->isEmpty("PRECOMPILED_DIR")) {
195 pchBaseName = Option::fixPathToTargetOS(project->first("PRECOMPILED_DIR"));
196 if(!pchBaseName.endsWith(Option::dir_sep))
197 pchBaseName += Option::dir_sep;
198 }
199 pchBaseName += project->first("QMAKE_ORIG_TARGET");
200
201 // replace place holders
202 pchFlags = pchFlags.replace("${QMAKE_PCH_INPUT}",
203 fileFixify(project->first("PRECOMPILED_HEADER")));
204 pchFlags = pchFlags.replace("${QMAKE_PCH_OUTPUT_BASE}", pchBaseName);
205 if (project->isActiveConfig("icc_pch_style")) {
206 // icc style
207 pchFlags = pchFlags.replace("${QMAKE_PCH_OUTPUT}",
208 pchBaseName + project->first("QMAKE_PCH_OUTPUT_EXT"));
209 }
210
211 if (!pchFlags.isEmpty())
212 compile_flag += " " + pchFlags;
213 }
214
215 QString cflags;
216 if(comps[i] == "OBJC" || comps[i] == "OBJCXX")
217 cflags += " $(CFLAGS)";
218 else
219 cflags += " $(" + comps[i] + "FLAGS)";
220 compile_flag += cflags + " $(INCPATH)";
221
222 QString compiler = comps[i];
223 if (compiler == "C")
224 compiler = "CC";
225
226 QString runComp = "QMAKE_RUN_" + compiler;
227 if(project->isEmpty(runComp))
228 project->values(runComp).append("$(" + compiler + ") " + compile_flag + " -o $obj $src");
229 QString runCompImp = "QMAKE_RUN_" + compiler + "_IMP";
230 if(project->isEmpty(runCompImp))
231 project->values(runCompImp).append("$(" + compiler + ") " + compile_flag + " -o \"$@\" \"$<\"");
232 }
233
234 if(project->isActiveConfig("macx") && !project->isEmpty("TARGET") && !project->isActiveConfig("compile_libtool") &&
235 ((project->isActiveConfig("build_pass") || project->isEmpty("BUILDS")))) {
236 QString bundle;
237 if(project->isActiveConfig("bundle") && !project->isEmpty("QMAKE_BUNDLE_EXTENSION")) {
238 bundle = unescapeFilePath(project->first("TARGET"));
239 if(!project->isEmpty("QMAKE_BUNDLE_NAME"))
240 bundle = unescapeFilePath(project->first("QMAKE_BUNDLE_NAME"));
241 if(!bundle.endsWith(project->first("QMAKE_BUNDLE_EXTENSION")))
242 bundle += project->first("QMAKE_BUNDLE_EXTENSION");
243 } else if(project->first("TEMPLATE") == "app" && project->isActiveConfig("app_bundle")) {
244 bundle = unescapeFilePath(project->first("TARGET"));
245 if(!project->isEmpty("QMAKE_APPLICATION_BUNDLE_NAME"))
246 bundle = unescapeFilePath(project->first("QMAKE_APPLICATION_BUNDLE_NAME"));
247 if(!bundle.endsWith(".app"))
248 bundle += ".app";
249 if(project->isEmpty("QMAKE_BUNDLE_LOCATION"))
250 project->values("QMAKE_BUNDLE_LOCATION").append("Contents/MacOS");
251 project->values("QMAKE_PKGINFO").append(project->first("DESTDIR") + bundle + "/Contents/PkgInfo");
252 project->values("QMAKE_BUNDLE_RESOURCE_FILE").append(project->first("DESTDIR") + bundle + "/Contents/Resources/empty.lproj");
253 } else if(project->first("TEMPLATE") == "lib" && !project->isActiveConfig("staticlib") &&
254 ((!project->isActiveConfig("plugin") && project->isActiveConfig("lib_bundle")) ||
255 (project->isActiveConfig("plugin") && project->isActiveConfig("plugin_bundle")))) {
256 bundle = unescapeFilePath(project->first("TARGET"));
257 if(project->isActiveConfig("plugin")) {
258 if(!project->isEmpty("QMAKE_PLUGIN_BUNDLE_NAME"))
259 bundle = unescapeFilePath(project->first("QMAKE_PLUGIN_BUNDLE_NAME"));
260 if(!project->isEmpty("QMAKE_BUNDLE_EXTENSION") && !bundle.endsWith(project->first("QMAKE_BUNDLE_EXTENSION")))
261 bundle += project->first("QMAKE_BUNDLE_EXTENSION");
262 else if(!bundle.endsWith(".plugin"))
263 bundle += ".plugin";
264 if(project->isEmpty("QMAKE_BUNDLE_LOCATION"))
265 project->values("QMAKE_BUNDLE_LOCATION").append("Contents/MacOS");
266 } else {
267 if(!project->isEmpty("QMAKE_FRAMEWORK_BUNDLE_NAME"))
268 bundle = unescapeFilePath(project->first("QMAKE_FRAMEWORK_BUNDLE_NAME"));
269 if(!project->isEmpty("QMAKE_BUNDLE_EXTENSION") && !bundle.endsWith(project->first("QMAKE_BUNDLE_EXTENSION")))
270 bundle += project->first("QMAKE_BUNDLE_EXTENSION");
271 else if(!bundle.endsWith(".framework"))
272 bundle += ".framework";
273 }
274 }
275 if(!bundle.isEmpty()) {
276 project->values("QMAKE_BUNDLE") = QStringList(bundle);
277 project->values("ALL_DEPS") += project->first("QMAKE_PKGINFO");
278 project->values("ALL_DEPS") += project->first("QMAKE_BUNDLE_RESOURCE_FILE");
279 } else {
280 project->values("QMAKE_BUNDLE").clear();
281 project->values("QMAKE_BUNDLE_LOCATION").clear();
282 }
283 } else { //no bundling here
284 project->values("QMAKE_BUNDLE").clear();
285 project->values("QMAKE_BUNDLE_LOCATION").clear();
286 }
287
288 if(!project->isEmpty("QMAKE_INTERNAL_INCLUDED_FILES"))
289 project->values("DISTFILES") += project->values("QMAKE_INTERNAL_INCLUDED_FILES");
290 project->values("DISTFILES") += project->projectFile();
291
292 init2();
293 project->values("QMAKE_INTERNAL_PRL_LIBS") << "QMAKE_LIBDIR_FLAGS" << "QMAKE_FRAMEWORKPATH_FLAGS" << "QMAKE_LIBS";
294 if(!project->isEmpty("QMAKE_MAX_FILES_PER_AR")) {
295 bool ok;
296 int max_files = project->first("QMAKE_MAX_FILES_PER_AR").toInt(&ok);
297 QStringList ar_sublibs, objs = project->values("OBJECTS");
298 if(ok && max_files > 5 && max_files < (int)objs.count()) {
299 QString lib;
300 for(int i = 0, obj_cnt = 0, lib_cnt = 0; i != objs.size(); ++i) {
301 if((++obj_cnt) >= max_files) {
302 if(lib_cnt) {
303 lib.sprintf("lib%s-tmp%d.a",
304 project->first("QMAKE_ORIG_TARGET").toLatin1().constData(), lib_cnt);
305 ar_sublibs << lib;
306 obj_cnt = 0;
307 }
308 lib_cnt++;
309 }
310 }
311 }
312 if(!ar_sublibs.isEmpty()) {
313 project->values("QMAKE_AR_SUBLIBS") = ar_sublibs;
314 project->values("QMAKE_INTERNAL_PRL_LIBS") << "QMAKE_AR_SUBLIBS";
315 }
316 }
317
318 if(project->isActiveConfig("compile_libtool")) {
319 const QString libtoolify[] = { "QMAKE_RUN_CC", "QMAKE_RUN_CC_IMP",
320 "QMAKE_RUN_CXX", "QMAKE_RUN_CXX_IMP",
321 "QMAKE_LINK_THREAD", "QMAKE_LINK", "QMAKE_AR_CMD", "QMAKE_LINK_SHLIB_CMD",
322 QString() };
323 for(int i = 0; !libtoolify[i].isNull(); i++) {
324 QStringList &l = project->values(libtoolify[i]);
325 if(!l.isEmpty()) {
326 QString libtool_flags, comp_flags;
327 if(libtoolify[i].startsWith("QMAKE_LINK") || libtoolify[i] == "QMAKE_AR_CMD") {
328 libtool_flags += " --mode=link";
329 if(project->isActiveConfig("staticlib")) {
330 libtool_flags += " -static";
331 } else {
332 if(!project->isEmpty("QMAKE_LIB_FLAG")) {
333 int maj = project->first("VER_MAJ").toInt();
334 int min = project->first("VER_MIN").toInt();
335 int pat = project->first("VER_PAT").toInt();
336 comp_flags += " -version-info " + QString::number(10*maj + min) +
337 ":" + QString::number(pat) + ":0";
338 if(libtoolify[i] != "QMAKE_AR_CMD") {
339 QString rpath = Option::output_dir;
340 if(!project->isEmpty("DESTDIR")) {
341 rpath = project->first("DESTDIR");
342 if(QDir::isRelativePath(rpath))
343 rpath.prepend(Option::output_dir + Option::dir_sep);
344 }
345 comp_flags += " -rpath " + Option::fixPathToTargetOS(rpath, false);
346 }
347 }
348 }
349 if(project->isActiveConfig("plugin"))
350 libtool_flags += " -module";
351 } else {
352 libtool_flags += " --mode=compile";
353 }
354 l.first().prepend("$(LIBTOOL)" + libtool_flags + " ");
355 if(!comp_flags.isEmpty())
356 l.first() += comp_flags;
357 }
358 }
359 }
360}
361
362void
363UnixMakefileGenerator::processPrlVariable(const QString &var, const QStringList &l)
364{
365 if(var == "QMAKE_PRL_LIBS") {
366 project->values("QMAKE_CURRENT_PRL_LIBS") += l;
367 } else
368 MakefileGenerator::processPrlVariable(var, l);
369}
370
371QStringList
372&UnixMakefileGenerator::findDependencies(const QString &file)
373{
374 QStringList &ret = MakefileGenerator::findDependencies(file);
375 // Note: The QMAKE_IMAGE_COLLECTION file have all images
376 // as dependency, so don't add precompiled header then
377 if(doPrecompiledHeaders() && !project->isEmpty("PRECOMPILED_HEADER")
378 && file != project->first("QMAKE_IMAGE_COLLECTION")) {
379 QString header_prefix;
380 if(!project->isEmpty("PRECOMPILED_DIR"))
381 header_prefix = project->first("PRECOMPILED_DIR");
382 header_prefix += project->first("QMAKE_ORIG_TARGET") + project->first("QMAKE_PCH_OUTPUT_EXT");
383 if (project->isActiveConfig("icc_pch_style")) {
384 // icc style
385 for(QStringList::Iterator it = Option::cpp_ext.begin(); it != Option::cpp_ext.end(); ++it) {
386 if(file.endsWith(*it)) {
387 ret += header_prefix;
388 break;
389 }
390 }
391 } else {
392 // gcc style
393 header_prefix += Option::dir_sep + project->first("QMAKE_PRECOMP_PREFIX");
394 for(QStringList::Iterator it = Option::c_ext.begin(); it != Option::c_ext.end(); ++it) {
395 if(file.endsWith(*it)) {
396 if(!project->isEmpty("QMAKE_CFLAGS_PRECOMPILE")) {
397 QString precomp_c_h = header_prefix + "c";
398 if(!ret.contains(precomp_c_h))
399 ret += precomp_c_h;
400 }
401 if(project->isActiveConfig("objective_c")) {
402 if(!project->isEmpty("QMAKE_OBJCFLAGS_PRECOMPILE")) {
403 QString precomp_objc_h = header_prefix + "objective-c";
404 if(!ret.contains(precomp_objc_h))
405 ret += precomp_objc_h;
406 }
407 if(!project->isEmpty("QMAKE_OBJCXXFLAGS_PRECOMPILE")) {
408 QString precomp_objcpp_h = header_prefix + "objective-c++";
409 if(!ret.contains(precomp_objcpp_h))
410 ret += precomp_objcpp_h;
411 }
412 }
413 break;
414 }
415 }
416 for(QStringList::Iterator it = Option::cpp_ext.begin(); it != Option::cpp_ext.end(); ++it) {
417 if(file.endsWith(*it)) {
418 if(!project->isEmpty("QMAKE_CXXFLAGS_PRECOMPILE")) {
419 QString precomp_cpp_h = header_prefix + "c++";
420 if(!ret.contains(precomp_cpp_h))
421 ret += precomp_cpp_h;
422 }
423 if(project->isActiveConfig("objective_c")) {
424 if(!project->isEmpty("QMAKE_OBJCXXFLAGS_PRECOMPILE")) {
425 QString precomp_objcpp_h = header_prefix + "objective-c++";
426 if(!ret.contains(precomp_objcpp_h))
427 ret += precomp_objcpp_h;
428 }
429 }
430 break;
431 }
432 }
433 }
434 }
435 return ret;
436}
437
438bool
439UnixMakefileGenerator::findLibraries()
440{
441 QList<QMakeLocalFileName> libdirs, frameworkdirs;
442 frameworkdirs.append(QMakeLocalFileName("/System/Library/Frameworks"));
443 frameworkdirs.append(QMakeLocalFileName("/Library/Frameworks"));
444 const QString lflags[] = { "QMAKE_LIBDIR_FLAGS", "QMAKE_FRAMEWORKPATH_FLAGS", "QMAKE_LFLAGS", "QMAKE_LIBS", "QMAKE_LIBS_PRIVATE", QString() };
445 for(int i = 0; !lflags[i].isNull(); i++) {
446 QStringList &l = project->values(lflags[i]);
447 for(QStringList::Iterator it = l.begin(); it != l.end(); ++it) {
448 bool do_suffix = true;
449 QString stub, dir, extn, opt = (*it).trimmed();
450 if(opt.startsWith("-")) {
451 if(opt.startsWith("-L")) {
452 QMakeLocalFileName f(opt.right(opt.length()-2));
453 if(!libdirs.contains(f))
454 libdirs.append(f);
455 } else if(opt.startsWith("-l")) {
456 if (!project->isEmpty("QMAKE_RVCT_LINKSTYLE")) {
457 (*it) = opt.mid(2);
458 } else if (project->isActiveConfig("rvct_linker")) {
459 (*it) = "lib" + opt.mid(2) + ".so";
460 } else {
461 stub = opt.mid(2);
462 }
463 } else if(Option::target_mode == Option::TARG_MACX_MODE && opt.startsWith("-F")) {
464 frameworkdirs.append(QMakeLocalFileName(opt.right(opt.length()-2)));
465 } else if(Option::target_mode == Option::TARG_MACX_MODE && opt.startsWith("-framework")) {
466 if(opt.length() > 11) {
467 opt = opt.mid(11);
468 } else {
469 ++it;
470 opt = (*it);
471 }
472 do_suffix = false;
473 extn = "";
474 dir = "/System/Library/Frameworks/" + opt + ".framework/";
475 stub = opt;
476 }
477 } else {
478 extn = dir = "";
479 stub = opt;
480 int slsh = opt.lastIndexOf(Option::dir_sep);
481 if(slsh != -1) {
482 dir = opt.left(slsh);
483 stub = opt.mid(slsh+1);
484 }
485 QRegExp stub_reg("^.*lib(" + stub + "[^./=]*)\\.(.*)$");
486 if(stub_reg.exactMatch(stub)) {
487 stub = stub_reg.cap(1);
488 extn = stub_reg.cap(2);
489 }
490 }
491 if(!stub.isEmpty()) {
492 if(do_suffix && !project->isEmpty("QMAKE_" + stub.toUpper() + "_SUFFIX"))
493 stub += project->first("QMAKE_" + stub.toUpper() + "_SUFFIX");
494 bool found = false;
495 QStringList extens;
496 if(!extn.isNull())
497 extens << extn;
498 else if (!project->isEmpty("QMAKE_SYMBIAN_SHLIB"))
499 // In Symbian you link to the stub .lib file, but run with the .dll file.
500 extens << "lib";
501 else
502 extens << project->values("QMAKE_EXTENSION_SHLIB").first() << "a";
503 for(QStringList::Iterator extit = extens.begin(); extit != extens.end(); ++extit) {
504 if(dir.isNull()) {
505 for(QList<QMakeLocalFileName>::Iterator dep_it = libdirs.begin(); dep_it != libdirs.end(); ++dep_it) {
506 QString pathToLib = ((*dep_it).local() + Option::dir_sep
507 + project->values("QMAKE_PREFIX_SHLIB").first()
508 + stub + "." + (*extit));
509 if(exists(pathToLib)) {
510 if (!project->isEmpty("QMAKE_RVCT_LINKSTYLE"))
511 (*it) = pathToLib;
512 else
513 (*it) = "-l" + stub;
514 found = true;
515 break;
516 }
517 }
518 } else {
519 if(exists(project->values("QMAKE_PREFIX_SHLIB").first() + stub + "." + (*extit))) {
520 (*it) = project->values("QMAKE_PREFIX_SHLIB").first() + stub + "." + (*extit);
521 found = true;
522 break;
523 }
524 }
525 }
526 if(!found && project->isActiveConfig("compile_libtool")) {
527 for(int dep_i = 0; dep_i < libdirs.size(); ++dep_i) {
528 if(exists(libdirs[dep_i].local() + Option::dir_sep + project->values("QMAKE_PREFIX_SHLIB").first() + stub + Option::libtool_ext)) {
529 (*it) = libdirs[dep_i].real() + Option::dir_sep + project->values("QMAKE_PREFIX_SHLIB").first() + stub + Option::libtool_ext;
530 found = true;
531 break;
532 }
533 }
534 }
535 }
536 }
537 }
538 return false;
539}
540
541QString linkLib(const QString &file, const QString &libName) {
542 QString ret;
543 QRegExp reg("^.*lib(" + QRegExp::escape(libName) + "[^./=]*).*$");
544 if(reg.exactMatch(file))
545 ret = "-l" + reg.cap(1);
546 return ret;
547}
548
549void
550UnixMakefileGenerator::processPrlFiles()
551{
552 QList<QMakeLocalFileName> libdirs, frameworkdirs;
553 frameworkdirs.append(QMakeLocalFileName("/System/Library/Frameworks"));
554 const QString lflags[] = { "QMAKE_LIBDIR_FLAGS", "QMAKE_FRAMEWORKPATH_FLAGS", "QMAKE_LFLAGS", "QMAKE_LIBS", QString() };
555 for(int i = 0; !lflags[i].isNull(); i++) {
556 QStringList &l = project->values(lflags[i]);
557 for(int lit = 0; lit < l.size(); ++lit) {
558 QString opt = l.at(lit).trimmed();
559 if(opt.startsWith("-")) {
560 if(opt.startsWith("-L")) {
561 QMakeLocalFileName l(opt.right(opt.length()-2));
562 if(!libdirs.contains(l))
563 libdirs.append(l);
564 } else if(opt.startsWith("-l")) {
565 QString lib = opt.right(opt.length() - 2);
566 for(int dep_i = 0; dep_i < libdirs.size(); ++dep_i) {
567 const QMakeLocalFileName &lfn = libdirs[dep_i];
568 if(!project->isActiveConfig("compile_libtool")) { //give them the .libs..
569 QString la = lfn.local() + Option::dir_sep + project->values("QMAKE_PREFIX_SHLIB").first() + lib + Option::libtool_ext;
570 if(exists(la) && QFile::exists(lfn.local() + Option::dir_sep + ".libs")) {
571 QString dot_libs = lfn.real() + Option::dir_sep + ".libs";
572 l.append("-L" + dot_libs);
573 libdirs.append(QMakeLocalFileName(dot_libs));
574 }
575 }
576
577 QString prl = lfn.local() + Option::dir_sep + project->values("QMAKE_PREFIX_SHLIB").first() + lib;
578 if(!project->isEmpty("QMAKE_" + lib.toUpper() + "_SUFFIX"))
579 prl += project->first("QMAKE_" + lib.toUpper() + "_SUFFIX");
580 if(processPrlFile(prl)) {
581 if(prl.startsWith(lfn.local()))
582 prl.replace(0, lfn.local().length(), lfn.real());
583 opt = linkLib(prl, lib);
584 break;
585 }
586 }
587 } else if(Option::target_mode == Option::TARG_MACX_MODE && opt.startsWith("-F")) {
588 QMakeLocalFileName f(opt.right(opt.length()-2));
589 if(!frameworkdirs.contains(f))
590 frameworkdirs.append(f);
591 } else if(Option::target_mode == Option::TARG_MACX_MODE && opt.startsWith("-framework")) {
592 if(opt.length() > 11)
593 opt = opt.mid(11);
594 else
595 opt = l.at(++lit);
596 opt = opt.trimmed();
597 const QList<QMakeLocalFileName> dirs = frameworkdirs + libdirs;
598 for(int dep_i = 0; dep_i < dirs.size(); ++dep_i) {
599 QString prl = dirs[dep_i].local() + "/" + opt + ".framework/" + opt + Option::prl_ext;
600 if(processPrlFile(prl))
601 break;
602 }
603 }
604 } else if(!opt.isNull()) {
605 QString lib = opt;
606 processPrlFile(lib);
607#if 0
608 if(ret)
609 opt = linkLib(lib, "");
610#endif
611 if(!opt.isEmpty())
612 l.replaceInStrings(lib, opt);
613 }
614
615 QStringList &prl_libs = project->values("QMAKE_CURRENT_PRL_LIBS");
616 if(!prl_libs.isEmpty()) {
617 for(int prl = 0; prl < prl_libs.size(); ++prl)
618 l.insert(lit+prl+1, prl_libs.at(prl));
619 prl_libs.clear();
620 }
621 }
622
623 //merge them into a logical order
624 if(!project->isActiveConfig("no_smart_library_merge") && !project->isActiveConfig("no_lflags_merge")) {
625 QHash<QString, QStringList> lflags;
626 for(int lit = 0; lit < l.size(); ++lit) {
627 QString arch("default");
628 QString opt = l.at(lit).trimmed();
629 if(opt.startsWith("-")) {
630 if (Option::target_mode == Option::TARG_MACX_MODE && opt.startsWith("-Xarch")) {
631 if (opt.length() > 7) {
632 arch = opt.mid(7);
633 opt = l.at(++lit);
634 }
635 }
636
637 if(opt.startsWith("-L") ||
638 (Option::target_mode == Option::TARG_MACX_MODE && opt.startsWith("-F"))) {
639 if(!lflags[arch].contains(opt))
640 lflags[arch].append(opt);
641 } else if(opt.startsWith("-l") || opt == "-pthread") {
642 // Make sure we keep the dependency-order of libraries
643 if (lflags[arch].contains(opt))
644 lflags[arch].removeAll(opt);
645 lflags[arch].append(opt);
646 } else if(Option::target_mode == Option::TARG_MACX_MODE && opt.startsWith("-framework")) {
647 if(opt.length() > 11)
648 opt = opt.mid(11);
649 else {
650 opt = l.at(++lit);
651 if (Option::target_mode == Option::TARG_MACX_MODE && opt.startsWith("-Xarch"))
652 opt = l.at(++lit); // The user has done the right thing and prefixed each part
653 }
654 bool found = false;
655 for(int x = 0; x < lflags[arch].size(); ++x) {
656 QString xf = lflags[arch].at(x);
657 if(xf.startsWith("-framework")) {
658 QString framework;
659 if(xf.length() > 11)
660 framework = xf.mid(11);
661 else
662 framework = lflags[arch].at(++x);
663 if(framework == opt) {
664 found = true;
665 break;
666 }
667 }
668 }
669 if(!found) {
670 lflags[arch].append("-framework");
671 lflags[arch].append(opt);
672 }
673 } else {
674 lflags[arch].append(opt);
675 }
676 } else if(!opt.isNull()) {
677 if(!lflags[arch].contains(opt))
678 lflags[arch].append(opt);
679 }
680 }
681
682 l = lflags.take("default");
683
684 // Process architecture specific options (Xarch)
685 QHash<QString, QStringList>::const_iterator archIterator = lflags.constBegin();
686 while (archIterator != lflags.constEnd()) {
687 const QStringList archOptions = archIterator.value();
688 for (int i = 0; i < archOptions.size(); ++i) {
689 l.append(QLatin1String("-Xarch_") + archIterator.key());
690 l.append(archOptions.at(i));
691 }
692 ++archIterator;
693 }
694 }
695 }
696}
697
698QString
699UnixMakefileGenerator::defaultInstall(const QString &t)
700{
701 if(t != "target" || project->first("TEMPLATE") == "subdirs")
702 return QString();
703
704 bool bundle = false;
705 const QString root = "$(INSTALL_ROOT)";
706 QStringList &uninst = project->values(t + ".uninstall");
707 QString ret, destdir=project->first("DESTDIR");
708 QString targetdir = Option::fixPathToTargetOS(project->first("target.path"), false);
709 if(!destdir.isEmpty() && destdir.right(1) != Option::dir_sep)
710 destdir += Option::dir_sep;
711 targetdir = fileFixify(targetdir, FileFixifyAbsolute);
712 if(targetdir.right(1) != Option::dir_sep)
713 targetdir += Option::dir_sep;
714
715 QStringList links;
716 QString target="$(TARGET)";
717 QStringList &targets = project->values(t + ".targets");
718 if(!project->isEmpty("QMAKE_BUNDLE")) {
719 target = project->first("QMAKE_BUNDLE");
720 bundle = true;
721 } else if(project->first("TEMPLATE") == "app") {
722 target = "$(QMAKE_TARGET)";
723 } else if(project->first("TEMPLATE") == "lib") {
724 if(project->isEmpty("QMAKE_CYGWIN_SHLIB")) {
725 if(!project->isActiveConfig("staticlib") && !project->isActiveConfig("plugin")) {
726 if(project->isEmpty("QMAKE_HPUX_SHLIB")) {
727 links << "$(TARGET0)" << "$(TARGET1)" << "$(TARGET2)";
728 } else {
729 links << "$(TARGET0)";
730 }
731 }
732 }
733 }
734 for(int i = 0; i < targets.size(); ++i) {
735 QString src = targets.at(i),
736 dst = filePrefixRoot(root, targetdir + src.section('/', -1));
737 if(!ret.isEmpty())
738 ret += "\n\t";
739 ret += "-$(INSTALL_FILE) \"" + src + "\" \"" + dst + "\"";
740 if(!uninst.isEmpty())
741 uninst.append("\n\t");
742 uninst.append("-$(DEL_FILE) \"" + dst + "\"");
743 }
744
745 if(!bundle && project->isActiveConfig("compile_libtool")) {
746 QString src_targ = target;
747 if(src_targ == "$(TARGET)")
748 src_targ = "$(TARGETL)";
749 QString dst_dir = fileFixify(targetdir, FileFixifyAbsolute);
750 if(QDir::isRelativePath(dst_dir))
751 dst_dir = Option::fixPathToTargetOS(Option::output_dir + Option::dir_sep + dst_dir);
752 ret = "-$(LIBTOOL) --mode=install cp \"" + src_targ + "\" \"" + filePrefixRoot(root, dst_dir) + "\"";
753 uninst.append("-$(LIBTOOL) --mode=uninstall \"" + src_targ + "\"");
754 } else {
755 QString src_targ = target;
756 if(!destdir.isEmpty())
757 src_targ = Option::fixPathToTargetOS(destdir + target, false);
758 QString dst_targ = filePrefixRoot(root, fileFixify(targetdir + target, FileFixifyAbsolute));
759 if(bundle) {
760 if(!ret.isEmpty())
761 ret += "\n\t";
762 ret += "$(DEL_FILE) -r \"" + dst_targ + "\"\n\t";
763 }
764 if(!ret.isEmpty())
765 ret += "\n\t";
766
767 QString copy_cmd("-");
768 if (bundle)
769 copy_cmd += "$(INSTALL_DIR)";
770 else if (project->first("TEMPLATE") == "lib" && project->isActiveConfig("staticlib"))
771 copy_cmd += "$(INSTALL_FILE)";
772 else
773 copy_cmd += "$(INSTALL_PROGRAM)";
774 copy_cmd += " \"" + src_targ + "\" \"" + dst_targ + "\"";
775 if(project->first("TEMPLATE") == "lib" && !project->isActiveConfig("staticlib")
776 && project->values(t + ".CONFIG").indexOf("fix_rpath") != -1) {
777 if(!project->isEmpty("QMAKE_FIX_RPATH")) {
778 ret += copy_cmd;
779 ret += "\n\t-" + var("QMAKE_FIX_RPATH") + " \"" +
780 dst_targ + "\" \"" + dst_targ + "\"";
781 } else if(!project->isEmpty("QMAKE_LFLAGS_RPATH")) {
782 ret += "-$(LINK) $(LFLAGS) " + var("QMAKE_LFLAGS_RPATH") + targetdir + " -o \"" +
783 dst_targ + "\" $(OBJECTS) $(LIBS) $(OBJCOMP)";
784 } else {
785 ret += copy_cmd;
786 }
787 } else {
788 ret += copy_cmd;
789 }
790
791 if(project->first("TEMPLATE") == "lib" && project->isActiveConfig("staticlib")) {
792 if(!project->isEmpty("QMAKE_RANLIB"))
793 ret += QString("\n\t$(RANLIB) \"") + dst_targ + "\"";
794 } else if(!project->isActiveConfig("debug") && !project->isActiveConfig("nostrip") && !project->isEmpty("QMAKE_STRIP")) {
795 ret += "\n\t-$(STRIP)";
796 if(project->first("TEMPLATE") == "lib" && !project->isEmpty("QMAKE_STRIPFLAGS_LIB"))
797 ret += " " + var("QMAKE_STRIPFLAGS_LIB");
798 else if(project->first("TEMPLATE") == "app" && !project->isEmpty("QMAKE_STRIPFLAGS_APP"))
799 ret += " " + var("QMAKE_STRIPFLAGS_APP");
800 if(bundle)
801 ret = " \"" + dst_targ + "/Contents/MacOS/$(QMAKE_TARGET)\"";
802 else
803 ret += " \"" + dst_targ + "\"";
804 }
805 if(!uninst.isEmpty())
806 uninst.append("\n\t");
807 if(bundle)
808 uninst.append("-$(DEL_FILE) -r \"" + dst_targ + "\"");
809 else
810 uninst.append("-$(DEL_FILE) \"" + dst_targ + "\"");
811 if(!links.isEmpty()) {
812 for(int i = 0; i < links.size(); ++i) {
813 if(Option::target_mode == Option::TARG_UNIX_MODE ||
814 Option::target_mode == Option::TARG_MACX_MODE) {
815 QString link = Option::fixPathToTargetOS(destdir + links[i], false);
816 int lslash = link.lastIndexOf(Option::dir_sep);
817 if(lslash != -1)
818 link = link.right(link.length() - (lslash + 1));
819 QString dst_link = filePrefixRoot(root, fileFixify(targetdir + link, FileFixifyAbsolute));
820 ret += "\n\t-$(SYMLINK) \"$(TARGET)\" \"" + dst_link + "\"";
821 if(!uninst.isEmpty())
822 uninst.append("\n\t");
823 uninst.append("-$(DEL_FILE) \"" + dst_link + "\"");
824 }
825 }
826 }
827 }
828 if(project->first("TEMPLATE") == "lib") {
829 QStringList types;
830 types << "prl" << "libtool" << "pkgconfig";
831 for(int i = 0; i < types.size(); ++i) {
832 const QString type = types.at(i);
833 QString meta;
834 if(type == "prl" && project->isActiveConfig("create_prl") && !project->isActiveConfig("no_install_prl") &&
835 !project->isEmpty("QMAKE_INTERNAL_PRL_FILE"))
836 meta = prlFileName(false);
837 if(type == "libtool" && project->isActiveConfig("create_libtool") && !project->isActiveConfig("compile_libtool"))
838 meta = libtoolFileName(false);
839 if(type == "pkgconfig" && project->isActiveConfig("create_pc"))
840 meta = pkgConfigFileName(false);
841 if(!meta.isEmpty()) {
842 QString src_meta = meta;
843 if(!destdir.isEmpty())
844 src_meta = Option::fixPathToTargetOS(destdir + meta, false);
845 QString dst_meta = filePrefixRoot(root, fileFixify(targetdir + meta, FileFixifyAbsolute));
846 if(!uninst.isEmpty())
847 uninst.append("\n\t");
848 uninst.append("-$(DEL_FILE) \"" + dst_meta + "\"");
849 const QString replace_rule("QMAKE_" + type.toUpper() + "_INSTALL_REPLACE");
850 const QString dst_meta_dir = fileInfo(dst_meta).path();
851 if(!dst_meta_dir.isEmpty()) {
852 if(!ret.isEmpty())
853 ret += "\n\t";
854 ret += mkdir_p_asstring(dst_meta_dir, true);
855 }
856 QString install_meta = "$(INSTALL_FILE) \"" + src_meta + "\" \"" + dst_meta + "\"";
857 if(project->isEmpty(replace_rule) || project->isActiveConfig("no_sed_meta_install")) {
858 if(!ret.isEmpty())
859 ret += "\n\t";
860 ret += "-" + install_meta;
861 } else {
862 if(!ret.isEmpty())
863 ret += "\n\t";
864 ret += "-$(SED)";
865 QStringList replace_rules = project->values(replace_rule);
866 for(int r = 0; r < replace_rules.size(); ++r) {
867 const QString match = project->first(replace_rules.at(r) + ".match"),
868 replace = project->first(replace_rules.at(r) + ".replace");
869 if(!match.isEmpty() /*&& match != replace*/)
870 ret += " -e \"s," + match + "," + replace + ",g\"";
871 }
872 ret += " \"" + src_meta + "\" >\"" + dst_meta + "\"";
873 //ret += " || " + install_meta;
874 }
875 }
876 }
877 }
878 return ret;
879}
880
881QString
882UnixMakefileGenerator::escapeFilePath(const QString &path) const
883{
884 QString ret = path;
885 if(!ret.isEmpty()) {
886 ret = unescapeFilePath(ret).replace(QLatin1Char(' '), QLatin1String("\\ "));
887 debug_msg(2, "EscapeFilePath: %s -> %s", path.toLatin1().constData(), ret.toLatin1().constData());
888 }
889 return ret;
890}
891
892QT_END_NAMESPACE
Note: See TracBrowser for help on using the repository browser.