0
|
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 qmake application 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 "symmake.h"
|
|
43 |
#include "initprojectdeploy_symbian.h"
|
|
44 |
|
|
45 |
#include <qstring.h>
|
|
46 |
#include <qhash.h>
|
|
47 |
#include <qstringlist.h>
|
|
48 |
#include <qdir.h>
|
|
49 |
#include <qdatetime.h>
|
|
50 |
#include <stdlib.h>
|
|
51 |
#include <qdebug.h>
|
|
52 |
|
|
53 |
#define RESOURCE_DIRECTORY_MMP "/resource/apps"
|
|
54 |
#define RESOURCE_DIRECTORY_RESOURCE "\\\\resource\\\\apps\\\\"
|
|
55 |
#define REGISTRATION_RESOURCE_DIRECTORY_HW "/private/10003a3f/import/apps"
|
|
56 |
#define PLUGIN_COMMON_DEF_FILE_FOR_MMP "./plugin_common.def"
|
|
57 |
#define PLUGIN_COMMON_DEF_FILE_ACTUAL "plugin_commonU.def"
|
|
58 |
#define BLD_INF_FILENAME_LEN (sizeof(BLD_INF_FILENAME) - 1)
|
|
59 |
|
|
60 |
#define BLD_INF_RULES_BASE "BLD_INF_RULES."
|
|
61 |
#define BLD_INF_TAG_PLATFORMS "prj_platforms"
|
|
62 |
#define BLD_INF_TAG_MMPFILES "prj_mmpfiles"
|
|
63 |
#define BLD_INF_TAG_TESTMMPFILES "prj_testmmpfiles"
|
|
64 |
#define BLD_INF_TAG_EXTENSIONS "prj_extensions"
|
|
65 |
|
|
66 |
#define RSS_RULES "RSS_RULES"
|
|
67 |
#define RSS_RULES_BASE "RSS_RULES."
|
|
68 |
#define RSS_TAG_NBROFICONS "number_of_icons"
|
|
69 |
#define RSS_TAG_ICONFILE "icon_file"
|
|
70 |
|
|
71 |
#define MMP_TARGET "TARGET"
|
|
72 |
#define MMP_TARGETTYPE "TARGETTYPE"
|
|
73 |
#define MMP_SECUREID "SECUREID"
|
|
74 |
#define MMP_OPTION_CW "OPTION CW"
|
|
75 |
#define MMP_OPTION_ARMCC "OPTION ARMCC"
|
|
76 |
#define MMP_OPTION_GCCE "OPTION GCCE"
|
|
77 |
|
|
78 |
#define SIS_TARGET "sis"
|
|
79 |
#define OK_SIS_TARGET "ok_sis"
|
|
80 |
#define FAIL_SIS_NOPKG_TARGET "fail_sis_nopkg"
|
|
81 |
#define FAIL_SIS_NOCACHE_TARGET "fail_sis_nocache"
|
|
82 |
#define RESTORE_BUILD_TARGET "restore_build"
|
|
83 |
|
|
84 |
#define PRINT_FILE_CREATE_ERROR(filename) fprintf(stderr, "Error: Could not create '%s'\n", qPrintable(filename));
|
|
85 |
|
|
86 |
QString SymbianMakefileGenerator::fixPathForMmp(const QString& origPath, const QDir& parentDir)
|
|
87 |
{
|
|
88 |
static QString epocRootStr;
|
|
89 |
if (epocRootStr.isEmpty()) {
|
|
90 |
QFileInfo efi(epocRoot());
|
|
91 |
epocRootStr = efi.canonicalFilePath();
|
|
92 |
if (epocRootStr.isEmpty()) {
|
|
93 |
fprintf(stderr, "Unable to resolve epocRoot '%s' to real dir on current drive, defaulting to '/' for mmp paths\n", qPrintable(epocRoot()));
|
|
94 |
epocRootStr = "/";
|
|
95 |
}
|
|
96 |
if (!epocRootStr.endsWith("/"))
|
|
97 |
epocRootStr += "/";
|
|
98 |
|
|
99 |
epocRootStr += "epoc32/";
|
|
100 |
}
|
|
101 |
|
|
102 |
QString resultPath = origPath;
|
|
103 |
|
|
104 |
// Make it relative, unless it starts with "%epocroot%/epoc32/"
|
|
105 |
if (resultPath.startsWith(epocRootStr, Qt::CaseInsensitive)) {
|
|
106 |
resultPath.replace(epocRootStr, "/epoc32/", Qt::CaseInsensitive);
|
|
107 |
} else {
|
|
108 |
resultPath = parentDir.relativeFilePath(resultPath);
|
|
109 |
}
|
|
110 |
resultPath = QDir::cleanPath(resultPath);
|
|
111 |
|
|
112 |
if (resultPath.isEmpty())
|
|
113 |
resultPath = ".";
|
|
114 |
|
|
115 |
return resultPath;
|
|
116 |
}
|
|
117 |
|
|
118 |
QString SymbianMakefileGenerator::canonizePath(const QString& origPath)
|
|
119 |
{
|
|
120 |
// Since current path gets appended almost always anyway, use it as default
|
|
121 |
// for nonexisting paths.
|
|
122 |
static QString defaultPath;
|
|
123 |
if (defaultPath.isEmpty()) {
|
|
124 |
QFileInfo fi(".");
|
|
125 |
defaultPath = fi.canonicalFilePath();
|
|
126 |
}
|
|
127 |
|
|
128 |
// Prepend epocroot to any paths beginning with "/epoc32/"
|
|
129 |
QString resultPath = QDir::fromNativeSeparators(origPath);
|
|
130 |
QString tempPath(resultPath);
|
|
131 |
bool isEpoc = false;
|
|
132 |
if (resultPath.startsWith("/epoc32/", Qt::CaseInsensitive)) {
|
|
133 |
isEpoc = true;
|
|
134 |
resultPath = QDir::fromNativeSeparators(epocRoot()) + resultPath.mid(1);
|
|
135 |
}
|
|
136 |
|
|
137 |
QFileInfo fi(fileInfo(resultPath));
|
|
138 |
if(fi.isDir()) {
|
|
139 |
if (isEpoc)
|
|
140 |
resultPath = fi.absoluteFilePath();//canonicalFilePath();
|
|
141 |
else
|
|
142 |
resultPath = fi.canonicalFilePath();
|
|
143 |
} else {
|
|
144 |
if (isEpoc)
|
|
145 |
resultPath = fi.absolutePath();//canonicalPath();
|
|
146 |
else
|
|
147 |
resultPath = fi.canonicalPath();
|
|
148 |
}
|
|
149 |
//some fix for the not existed EPOC32\include folder
|
|
150 |
if (isEpoc) {
|
|
151 |
int index = resultPath.lastIndexOf("/epoc32/");
|
|
152 |
QString tmpRes = resultPath.mid(index);
|
|
153 |
if (tmpRes != tempPath) {
|
|
154 |
//we have the problems for not existed include directory
|
|
155 |
//change the path
|
|
156 |
resultPath.replace(tmpRes, tempPath);
|
|
157 |
}
|
|
158 |
}
|
|
159 |
|
|
160 |
resultPath = QDir::cleanPath(resultPath);
|
|
161 |
|
|
162 |
if (resultPath.isEmpty())
|
|
163 |
resultPath = defaultPath;
|
|
164 |
|
|
165 |
return resultPath;
|
|
166 |
}
|
|
167 |
|
|
168 |
SymbianMakefileGenerator::SymbianMakefileGenerator() : MakefileGenerator() { }
|
|
169 |
SymbianMakefileGenerator::~SymbianMakefileGenerator() { }
|
|
170 |
|
|
171 |
void SymbianMakefileGenerator::writeHeader(QTextStream &t)
|
|
172 |
{
|
|
173 |
t << "// ============================================================================" << endl;
|
|
174 |
t << "// * Makefile for building: " << escapeFilePath(var("TARGET")) << endl;
|
|
175 |
t << "// * Generated by qmake (" << qmake_version() << ") (Qt " << QT_VERSION_STR << ") on: ";
|
|
176 |
t << QDateTime::currentDateTime().toString(Qt::ISODate) << endl;
|
|
177 |
t << "// * This file is generated by qmake and should not be modified by the" << endl;
|
|
178 |
t << "// * user." << endl;
|
|
179 |
t << "// * Project: " << fileFixify(project->projectFile()) << endl;
|
|
180 |
t << "// * Template: " << var("TEMPLATE") << endl;
|
|
181 |
t << "// ============================================================================" << endl;
|
|
182 |
t << endl;
|
|
183 |
|
|
184 |
// Defining define for bld.inf
|
|
185 |
|
|
186 |
QString shortProFilename = project->projectFile();
|
|
187 |
shortProFilename.replace(0, shortProFilename.lastIndexOf("/") + 1, QString(""));
|
|
188 |
shortProFilename.replace(Option::pro_ext, QString(""));
|
|
189 |
|
|
190 |
QString bldinfDefine = shortProFilename;
|
|
191 |
bldinfDefine.append("_");
|
|
192 |
bldinfDefine.append(generate_uid(project->projectFile()));
|
|
193 |
|
|
194 |
bldinfDefine.prepend("BLD_INF_");
|
|
195 |
removeSpecialCharacters(bldinfDefine);
|
|
196 |
|
|
197 |
t << "#define " << bldinfDefine.toUpper() << endl << endl;
|
|
198 |
}
|
|
199 |
|
|
200 |
bool SymbianMakefileGenerator::writeMakefile(QTextStream &t)
|
|
201 |
{
|
|
202 |
writeHeader(t);
|
|
203 |
|
|
204 |
QString numberOfIcons;
|
|
205 |
QString iconFile;
|
|
206 |
QStringList userRssRules;
|
|
207 |
readRssRules(numberOfIcons, iconFile, userRssRules);
|
|
208 |
|
|
209 |
// Get the application translations and convert to symbian OS lang code, i.e. decical number
|
|
210 |
QStringList symbianLangCodes = symbianLangCodesFromTsFiles();
|
|
211 |
|
|
212 |
// Generate pkg files if there are any actual files to deploy
|
|
213 |
bool generatePkg = false;
|
|
214 |
if (targetType == TypeExe) {
|
|
215 |
generatePkg = true;
|
|
216 |
} else {
|
|
217 |
foreach(QString item, project->values("DEPLOYMENT")) {
|
|
218 |
if (!project->values(item + ".sources").isEmpty()) {
|
|
219 |
generatePkg = true;
|
|
220 |
break;
|
|
221 |
}
|
|
222 |
}
|
|
223 |
}
|
|
224 |
|
|
225 |
if (generatePkg) {
|
|
226 |
generatePkgFile(iconFile);
|
|
227 |
}
|
|
228 |
|
|
229 |
writeBldInfContent(t, generatePkg);
|
|
230 |
|
|
231 |
// Generate empty wrapper makefile here, because wrapper makefile must exist before writeMkFile,
|
|
232 |
// but all required data is not yet available.
|
|
233 |
bool isPrimaryMakefile = true;
|
|
234 |
QString wrapperFileName("Makefile");
|
|
235 |
QString outputFileName = fileInfo(Option::output.fileName()).fileName();
|
|
236 |
if (outputFileName != BLD_INF_FILENAME) {
|
|
237 |
wrapperFileName.append(".").append((outputFileName.size() > BLD_INF_FILENAME_LEN && outputFileName.left(BLD_INF_FILENAME_LEN) == BLD_INF_FILENAME) ? outputFileName.mid(8) : outputFileName);
|
|
238 |
isPrimaryMakefile = false;
|
|
239 |
}
|
|
240 |
|
|
241 |
QFile wrapperMakefile(wrapperFileName);
|
|
242 |
if (wrapperMakefile.open(QIODevice::WriteOnly)) {
|
|
243 |
generatedFiles << wrapperFileName;
|
|
244 |
if (Option::mkfile::listgen) {
|
|
245 |
generatePrint(fileInfo(wrapperMakefile.fileName()).absoluteFilePath());
|
|
246 |
}
|
|
247 |
|
|
248 |
} else {
|
|
249 |
PRINT_FILE_CREATE_ERROR(wrapperFileName);
|
|
250 |
return false;
|
|
251 |
}
|
|
252 |
|
|
253 |
if (targetType == TypeSubdirs) {
|
|
254 |
// If we have something to deploy, generate extension makefile for just that, since
|
|
255 |
// normal extension makefile is not getting generated and we need emulator deployment to be done.
|
|
256 |
if (generatePkg)
|
|
257 |
writeMkFile(wrapperFileName, true);
|
|
258 |
writeWrapperMakefile(wrapperMakefile, isPrimaryMakefile);
|
|
259 |
return true;
|
|
260 |
}
|
|
261 |
|
|
262 |
writeMkFile(wrapperFileName, false);
|
|
263 |
|
|
264 |
QString shortProFilename = project->projectFile();
|
|
265 |
shortProFilename.replace(0, shortProFilename.lastIndexOf("/") + 1, QString(""));
|
|
266 |
shortProFilename.replace(Option::pro_ext, QString(""));
|
|
267 |
|
|
268 |
QString mmpFilename = shortProFilename;
|
|
269 |
mmpFilename.append("_");
|
|
270 |
mmpFilename.append(uid3);
|
|
271 |
mmpFilename.append(Option::mmp_ext);
|
|
272 |
writeMmpFile(mmpFilename, symbianLangCodes);
|
|
273 |
|
|
274 |
if (targetType == TypeExe) {
|
|
275 |
if (!project->values("CONFIG").contains("no_icon", Qt::CaseInsensitive)) {
|
|
276 |
writeRegRssFile(userRssRules);
|
|
277 |
writeRssFile(numberOfIcons, iconFile);
|
|
278 |
writeLocFile(symbianLangCodes);
|
|
279 |
}
|
|
280 |
}
|
|
281 |
|
|
282 |
writeCustomDefFile();
|
|
283 |
writeWrapperMakefile(wrapperMakefile, isPrimaryMakefile);
|
|
284 |
|
|
285 |
return true;
|
|
286 |
}
|
|
287 |
|
|
288 |
void SymbianMakefileGenerator::generatePkgFile(const QString &iconFile)
|
|
289 |
{
|
|
290 |
QString pkgFilename = QString("%1_template.%2")
|
|
291 |
.arg(fixedTarget)
|
|
292 |
.arg("pkg");
|
|
293 |
QFile pkgFile(pkgFilename);
|
|
294 |
if (!pkgFile.open(QIODevice::WriteOnly | QIODevice::Text)) {
|
|
295 |
PRINT_FILE_CREATE_ERROR(pkgFilename);
|
|
296 |
return;
|
|
297 |
}
|
|
298 |
if (Option::mkfile::listgen) {
|
|
299 |
generatePrint(fileInfo(pkgFile.fileName()).absoluteFilePath());
|
|
300 |
}
|
|
301 |
generatedFiles << pkgFile.fileName();
|
|
302 |
|
|
303 |
// Header info
|
|
304 |
QTextStream t(&pkgFile);
|
|
305 |
t << QString("; %1 generated by qmake at %2").arg(pkgFilename).arg(QDateTime::currentDateTime().toString(Qt::ISODate)) << endl;
|
|
306 |
t << "; This file is generated by qmake and should not be modified by the user" << endl;
|
|
307 |
t << ";" << endl << endl;
|
|
308 |
|
|
309 |
// Construct QStringList from pkg_prerules since we need search it before printed to file
|
|
310 |
QStringList rawPkgPreRules;
|
|
311 |
foreach(QString deploymentItem, project->values("DEPLOYMENT")) {
|
|
312 |
foreach(QString pkgrulesItem, project->values(deploymentItem + ".pkg_prerules")) {
|
|
313 |
QStringList pkgrulesValue = project->values(pkgrulesItem);
|
|
314 |
// If there is no stringlist defined for a rule, use rule name directly
|
|
315 |
// This is convenience for defining single line mmp statements
|
|
316 |
if (pkgrulesValue.isEmpty()) {
|
|
317 |
rawPkgPreRules << pkgrulesItem;
|
|
318 |
} else {
|
|
319 |
foreach(QString pkgrule, pkgrulesValue) {
|
|
320 |
rawPkgPreRules << pkgrule;
|
|
321 |
}
|
|
322 |
}
|
|
323 |
}
|
|
324 |
}
|
|
325 |
|
|
326 |
// Apply some defaults if specific data does not exist in PKG pre-rules
|
|
327 |
|
|
328 |
if (!containsStartWithItem('&', rawPkgPreRules)) {
|
|
329 |
// language, (*** hardcoded to english atm, should be parsed from TRANSLATIONS)
|
|
330 |
t << "; Language" << endl;
|
|
331 |
t << "&EN" << endl << endl;
|
|
332 |
} else {
|
|
333 |
// In case user defines langs, he must take care also about SIS header
|
|
334 |
if (!containsStartWithItem('#', rawPkgPreRules))
|
|
335 |
fprintf(stderr, "Warning: If language is defined with DEPLOYMENT pkg_prerules, also the SIS header must be defined\n");
|
|
336 |
}
|
|
337 |
|
|
338 |
// name of application, UID and version
|
|
339 |
QString applicationVersion = project->first("VERSION").isEmpty() ? "1,0,0" : project->first("VERSION").replace('.', ',');
|
|
340 |
|
|
341 |
if (!containsStartWithItem('#', rawPkgPreRules)) {
|
|
342 |
QString visualTarget = escapeFilePath(fileFixify(project->first("TARGET")));
|
|
343 |
visualTarget = removePathSeparators(visualTarget);
|
|
344 |
|
|
345 |
t << "; SIS header: name, uid, version" << endl;
|
|
346 |
t << QString("#{\"%1\"},(%2),%3").arg(visualTarget).arg(uid3).arg(applicationVersion) << endl << endl;
|
|
347 |
}
|
|
348 |
|
|
349 |
// Localized vendor name
|
|
350 |
if (!containsStartWithItem('%', rawPkgPreRules)) {
|
|
351 |
t << "; Localised Vendor name" << endl;
|
|
352 |
t << "%{\"Vendor\"}" << endl << endl;
|
|
353 |
}
|
|
354 |
|
|
355 |
// Unique vendor name
|
|
356 |
if (!containsStartWithItem(':', rawPkgPreRules)) {
|
|
357 |
t << "; Unique Vendor name" << endl;
|
|
358 |
t << ":\"Vendor\"" << endl << endl;
|
|
359 |
}
|
|
360 |
|
|
361 |
// PKG pre-rules - these are added before actual file installations i.e. SIS package body
|
|
362 |
if (rawPkgPreRules.size()) {
|
|
363 |
t << "; Manual PKG pre-rules from PRO files" << endl;
|
|
364 |
foreach(QString item, rawPkgPreRules) {
|
|
365 |
t << item << endl;
|
|
366 |
}
|
|
367 |
t << endl;
|
|
368 |
}
|
|
369 |
|
|
370 |
// Install paths on the phone *** should be dynamic at some point
|
|
371 |
QString installPathBin = "!:\\sys\\bin";
|
|
372 |
QString installPathResource = "!:\\resource\\apps";
|
|
373 |
QString installPathRegResource = "!:\\private\\10003a3f\\import\\apps";
|
|
374 |
|
|
375 |
// Find location of builds
|
|
376 |
QString epocReleasePath = QString("%1epoc32/release/$(PLATFORM)/$(TARGET)")
|
|
377 |
.arg(epocRoot());
|
|
378 |
|
|
379 |
|
|
380 |
if (targetType == TypeExe) {
|
|
381 |
// deploy .exe file
|
|
382 |
t << "; Executable and default resource files" << endl;
|
|
383 |
QString exeFile = fixedTarget + ".exe";
|
|
384 |
t << QString("\"%1/%2\" - \"%3\\%4\"")
|
|
385 |
.arg(epocReleasePath)
|
|
386 |
.arg(exeFile)
|
|
387 |
.arg(installPathBin)
|
|
388 |
.arg(exeFile) << endl;
|
|
389 |
|
|
390 |
// deploy rsc & reg_rsc file
|
|
391 |
if (!project->values("CONFIG").contains("no_icon", Qt::CaseInsensitive)) {
|
|
392 |
t << QString("\"%1epoc32/data/z/resource/apps/%2\" - \"%3\\%4\"")
|
|
393 |
.arg(epocRoot())
|
|
394 |
.arg(fixedTarget + ".rsc")
|
|
395 |
.arg(installPathResource)
|
|
396 |
.arg(fixedTarget + ".rsc") << endl;
|
|
397 |
|
|
398 |
t << QString("\"%1epoc32/data/z/private/10003a3f/import/apps/%2\" - \"%3\\%4\"")
|
|
399 |
.arg(epocRoot())
|
|
400 |
.arg(fixedTarget + "_reg.rsc")
|
|
401 |
.arg(installPathRegResource)
|
|
402 |
.arg(fixedTarget + "_reg.rsc") << endl;
|
|
403 |
|
|
404 |
QString myIconFile = iconFile;
|
|
405 |
myIconFile = myIconFile.replace("\\\\", "\\");
|
|
406 |
|
|
407 |
if (!iconFile.isEmpty()) {
|
|
408 |
t << QString("\"%1epoc32/data/z%2\" - \"!:%3\"")
|
|
409 |
.arg(epocRoot())
|
|
410 |
.arg(QString(myIconFile).replace('\\','/'))
|
|
411 |
.arg(myIconFile) << endl << endl;
|
|
412 |
}
|
|
413 |
}
|
|
414 |
}
|
|
415 |
|
|
416 |
// deploy any additional DEPLOYMENT files
|
|
417 |
DeploymentList depList;
|
|
418 |
QString remoteTestPath;
|
|
419 |
remoteTestPath = QString("!:\\private\\%1").arg(privateDirUid);
|
|
420 |
|
|
421 |
initProjectDeploySymbian(project, depList, remoteTestPath, true, "$(PLATFORM)", "$(TARGET)", generatedDirs, generatedFiles);
|
|
422 |
if (depList.size())
|
|
423 |
t << "; DEPLOYMENT" << endl;
|
|
424 |
for (int i = 0; i < depList.size(); ++i) {
|
|
425 |
t << QString("\"%1\" - \"%2\"")
|
|
426 |
.arg(QString(depList.at(i).from).replace('\\','/'))
|
|
427 |
.arg(depList.at(i).to) << endl;
|
|
428 |
}
|
|
429 |
t << endl;
|
|
430 |
|
|
431 |
// PKG post-rules - these are added after actual file installations i.e. SIS package body
|
|
432 |
t << "; Manual PKG post-rules from PRO files" << endl;
|
|
433 |
foreach(QString deploymentItem, project->values("DEPLOYMENT")) {
|
|
434 |
foreach(QString pkgrulesItem, project->values(deploymentItem + ".pkg_postrules")) {
|
|
435 |
QStringList pkgrulesValue = project->values(pkgrulesItem);
|
|
436 |
// If there is no stringlist defined for a rule, use rule name directly
|
|
437 |
// This is convenience for defining single line statements
|
|
438 |
if (pkgrulesValue.isEmpty()) {
|
|
439 |
t << pkgrulesItem << endl;
|
|
440 |
} else {
|
|
441 |
foreach(QString pkgrule, pkgrulesValue) {
|
|
442 |
t << pkgrule << endl;
|
|
443 |
}
|
|
444 |
}
|
|
445 |
t << endl;
|
|
446 |
}
|
|
447 |
}
|
|
448 |
}
|
|
449 |
|
|
450 |
bool SymbianMakefileGenerator::containsStartWithItem(const QChar &c, const QStringList& src)
|
|
451 |
{
|
|
452 |
bool result = false;
|
|
453 |
foreach(QString str, src) {
|
|
454 |
if (str.startsWith(c)) {
|
|
455 |
result = true;
|
|
456 |
break;
|
|
457 |
}
|
|
458 |
}
|
|
459 |
return result;
|
|
460 |
}
|
|
461 |
|
|
462 |
void SymbianMakefileGenerator::writeCustomDefFile()
|
|
463 |
{
|
|
464 |
if (targetType == TypePlugin && !project->values("CONFIG").contains("stdbinary", Qt::CaseInsensitive)) {
|
|
465 |
// Create custom def file for plugin
|
|
466 |
QFile ft(QLatin1String(PLUGIN_COMMON_DEF_FILE_ACTUAL));
|
|
467 |
|
|
468 |
if (ft.open(QIODevice::WriteOnly)) {
|
|
469 |
if (Option::mkfile::listgen) {
|
|
470 |
generatePrint(fileInfo(ft.fileName()).absoluteFilePath());
|
|
471 |
}
|
|
472 |
generatedFiles << ft.fileName();
|
|
473 |
QTextStream t(&ft);
|
|
474 |
|
|
475 |
t << "; ==============================================================================" << endl;
|
|
476 |
t << "; Generated by qmake (" << qmake_version() << ") (Qt " << QT_VERSION_STR << ") on: ";
|
|
477 |
t << QDateTime::currentDateTime().toString(Qt::ISODate) << endl;
|
|
478 |
t << "; This file is generated by qmake and should not be modified by the" << endl;
|
|
479 |
t << "; user." << endl;
|
|
480 |
t << "; Name : " PLUGIN_COMMON_DEF_FILE_ACTUAL << endl;
|
|
481 |
t << "; Part of : " << project->values("TARGET").join(" ") << endl;
|
|
482 |
t << "; Description : Fixes common plugin symbols to known ordinals" << endl;
|
|
483 |
t << "; Version : " << endl;
|
|
484 |
t << ";" << endl;
|
|
485 |
t << "; ==============================================================================" << "\n" << endl;
|
|
486 |
|
|
487 |
t << endl;
|
|
488 |
|
|
489 |
t << "EXPORTS" << endl;
|
|
490 |
t << "\tqt_plugin_query_verification_data @ 1 NONAME" << endl;
|
|
491 |
t << "\tqt_plugin_instance @ 2 NONAME" << endl;
|
|
492 |
t << endl;
|
|
493 |
} else {
|
|
494 |
PRINT_FILE_CREATE_ERROR(QString(PLUGIN_COMMON_DEF_FILE_ACTUAL))
|
|
495 |
}
|
|
496 |
}
|
|
497 |
}
|
|
498 |
|
|
499 |
void SymbianMakefileGenerator::init()
|
|
500 |
{
|
|
501 |
MakefileGenerator::init();
|
|
502 |
fixedTarget = escapeFilePath(fileFixify(project->first("TARGET")));
|
|
503 |
fixedTarget = removePathSeparators(fixedTarget);
|
|
504 |
removeSpecialCharacters(fixedTarget);
|
|
505 |
|
|
506 |
if (0 != project->values("QMAKE_PLATFORM").size())
|
|
507 |
platform = varGlue("QMAKE_PLATFORM", "", " ", "");
|
|
508 |
|
|
509 |
if (0 == project->values("QMAKESPEC").size())
|
|
510 |
project->values("QMAKESPEC").append(qgetenv("QMAKESPEC"));
|
|
511 |
|
|
512 |
project->values("QMAKE_LIBS") += escapeFilePaths(project->values("LIBS"));
|
|
513 |
project->values("QMAKE_LIBS_PRIVATE") += escapeFilePaths(project->values("LIBS_PRIVATE"));
|
|
514 |
|
|
515 |
// bld.inf
|
|
516 |
project->values("MAKEFILE") += BLD_INF_FILENAME;
|
|
517 |
|
|
518 |
// .mmp
|
|
519 |
initMmpVariables();
|
|
520 |
|
|
521 |
// Check TARGET.UID3 presence
|
|
522 |
if (0 != project->values("TARGET.UID3").size()) {
|
|
523 |
uid3 = project->first("TARGET.UID3");
|
|
524 |
} else {
|
|
525 |
uid3 = generateUID3();
|
|
526 |
}
|
|
527 |
|
|
528 |
if ((project->values("TEMPLATE")).contains("app"))
|
|
529 |
targetType = TypeExe;
|
|
530 |
else if ((project->values("TEMPLATE")).contains("lib")) {
|
|
531 |
// Check CONFIG to see if we are to build staticlib or dll
|
|
532 |
if (project->values("CONFIG").contains("staticlib") || project->values("CONFIG").contains("static"))
|
|
533 |
targetType = TypeLib;
|
|
534 |
else if (project->values("CONFIG").contains("plugin"))
|
|
535 |
targetType = TypePlugin;
|
|
536 |
else
|
|
537 |
targetType = TypeDll;
|
|
538 |
} else {
|
|
539 |
targetType = TypeSubdirs;
|
|
540 |
}
|
|
541 |
|
|
542 |
if (0 != project->values("TARGET.UID2").size()) {
|
|
543 |
uid2 = project->first("TARGET.UID2");
|
|
544 |
} else if (project->values("CONFIG").contains("stdbinary", Qt::CaseInsensitive)) {
|
|
545 |
uid2 = "0x20004C45";
|
|
546 |
} else {
|
|
547 |
if (targetType == TypeExe) {
|
|
548 |
if (project->values("QT").contains("gui", Qt::CaseInsensitive)) {
|
|
549 |
// exe and gui -> uid2 needed
|
|
550 |
uid2 = "0x100039CE";
|
|
551 |
} else {
|
|
552 |
// exe but not gui: uid2 is ignored anyway -> set it to 0
|
|
553 |
uid2 = "0";
|
|
554 |
}
|
|
555 |
} else if (targetType == TypeDll || targetType == TypeLib || targetType == TypePlugin) {
|
|
556 |
uid2 = "0x1000008d";
|
|
557 |
}
|
|
558 |
}
|
|
559 |
|
|
560 |
uid2 = uid2.trimmed();
|
|
561 |
uid3 = uid3.trimmed();
|
|
562 |
|
|
563 |
// UID is valid as either hex or decimal, so just convert it to number and back to hex
|
|
564 |
// to get proper string for private dir
|
|
565 |
bool conversionOk = false;
|
|
566 |
uint uidNum = uid3.toUInt(&conversionOk, 0);
|
|
567 |
|
|
568 |
if (!conversionOk) {
|
|
569 |
fprintf(stderr, "Error: Invalid UID \"%s\".\n", uid3.toUtf8().constData());
|
|
570 |
} else {
|
|
571 |
privateDirUid.setNum(uidNum, 16);
|
|
572 |
while (privateDirUid.length() < 8)
|
|
573 |
privateDirUid.insert(0, QLatin1Char('0'));
|
|
574 |
}
|
|
575 |
}
|
|
576 |
|
|
577 |
QString SymbianMakefileGenerator::getTargetExtension()
|
|
578 |
{
|
|
579 |
QString ret;
|
|
580 |
if (targetType == TypeExe) {
|
|
581 |
ret.append("exe");
|
|
582 |
} else if (targetType == TypeLib) {
|
|
583 |
ret.append("lib");
|
|
584 |
} else if (targetType == TypeDll || targetType == TypePlugin) {
|
|
585 |
ret.append("dll");
|
|
586 |
} else if (targetType == TypeSubdirs) {
|
|
587 |
// Not actually usable, so return empty
|
|
588 |
} else {
|
|
589 |
// If nothing else set, default to exe
|
|
590 |
ret.append("exe");
|
|
591 |
}
|
|
592 |
|
|
593 |
return ret;
|
|
594 |
}
|
|
595 |
|
|
596 |
QString SymbianMakefileGenerator::generateUID3()
|
|
597 |
{
|
|
598 |
QString target = project->first("TARGET");
|
|
599 |
QString currPath = qmake_getpwd();
|
|
600 |
target.prepend("/").prepend(currPath);
|
|
601 |
return generate_test_uid(target);
|
|
602 |
}
|
|
603 |
|
|
604 |
void SymbianMakefileGenerator::initMmpVariables()
|
|
605 |
{
|
|
606 |
QStringList sysincspaths;
|
|
607 |
QStringList srcincpaths;
|
|
608 |
QStringList srcpaths;
|
|
609 |
|
|
610 |
srcpaths << project->values("SOURCES") << project->values("GENERATED_SOURCES");
|
|
611 |
srcpaths << project->values("UNUSED_SOURCES") << project->values("UI_SOURCES_DIR");
|
|
612 |
srcpaths << project->values("UI_DIR");
|
|
613 |
|
|
614 |
QDir current = QDir::current();
|
|
615 |
QString canonizedCurrent = canonizePath(".");
|
|
616 |
|
|
617 |
for (int j = 0; j < srcpaths.size(); ++j) {
|
|
618 |
QFileInfo fi(fileInfo(srcpaths.at(j)));
|
|
619 |
// Sometimes sources have other than *.c* files (e.g. *.moc); prune them.
|
|
620 |
if (fi.suffix().startsWith("c")) {
|
|
621 |
if (fi.filePath().length() > fi.fileName().length()) {
|
|
622 |
appendIfnotExist(srcincpaths, fi.path());
|
|
623 |
sources[canonizePath(fi.path())] += fi.fileName();
|
|
624 |
} else {
|
|
625 |
sources[canonizedCurrent] += fi.fileName();
|
|
626 |
appendIfnotExist(srcincpaths, canonizedCurrent);
|
|
627 |
}
|
|
628 |
}
|
|
629 |
}
|
|
630 |
|
|
631 |
QStringList incpaths;
|
|
632 |
incpaths << project->values("INCLUDEPATH");
|
|
633 |
incpaths << QLibraryInfo::location(QLibraryInfo::HeadersPath);
|
|
634 |
incpaths << project->values("HEADERS");
|
|
635 |
incpaths << srcincpaths;
|
|
636 |
incpaths << project->values("UI_HEADERS_DIR");
|
|
637 |
incpaths << project->values("UI_DIR");
|
|
638 |
|
|
639 |
QString epocPath("epoc32");
|
|
640 |
for (int j = 0; j < incpaths.size(); ++j) {
|
|
641 |
QString includepath = canonizePath(incpaths.at(j));
|
|
642 |
appendIfnotExist(sysincspaths, includepath);
|
|
643 |
// As a workaround for Symbian toolchain insistence to treat include
|
|
644 |
// statements as relative to source file rather than the file they appear in,
|
|
645 |
// we generate extra temporary include directories to make
|
|
646 |
// relative include paths used in various headers to work properly.
|
|
647 |
// Note that this is not a fix-all solution; it's just a stop-gap measure
|
|
648 |
// to make Qt itself build until toolchain can support relative includes in
|
|
649 |
// a way that Qt expects.
|
|
650 |
if (!includepath.contains(epocPath)) // No temp dirs for epoc includes
|
|
651 |
appendIfnotExist(sysincspaths, includepath + QString("/" QT_EXTRA_INCLUDE_DIR));
|
|
652 |
}
|
|
653 |
|
|
654 |
// Remove duplicate include path entries
|
|
655 |
QStringList temporary;
|
|
656 |
for (int i = 0; i < sysincspaths.size(); ++i) {
|
|
657 |
QString origPath = sysincspaths.at(i);
|
|
658 |
QFileInfo origPathInfo(fileInfo(origPath));
|
|
659 |
bool bFound = false;
|
|
660 |
|
|
661 |
for (int j = 0; j < temporary.size(); ++j) {
|
|
662 |
QString tmpPath = temporary.at(j);
|
|
663 |
QFileInfo tmpPathInfo(fileInfo(tmpPath));
|
|
664 |
|
|
665 |
if (origPathInfo.absoluteFilePath() == tmpPathInfo.absoluteFilePath()) {
|
|
666 |
bFound = true;
|
|
667 |
if (!tmpPathInfo.isRelative() && origPathInfo.isRelative()) {
|
|
668 |
// We keep the relative notation
|
|
669 |
temporary.removeOne(tmpPath);
|
|
670 |
temporary << origPath;
|
|
671 |
}
|
|
672 |
}
|
|
673 |
}
|
|
674 |
|
|
675 |
if (!bFound)
|
|
676 |
temporary << origPath;
|
|
677 |
|
|
678 |
}
|
|
679 |
|
|
680 |
sysincspaths.clear();
|
|
681 |
sysincspaths << temporary;
|
|
682 |
|
|
683 |
systeminclude.insert("SYSTEMINCLUDE", sysincspaths);
|
|
684 |
|
|
685 |
// Check MMP_RULES for singleton keywords that are overridden
|
|
686 |
QStringList overridableMmpKeywords;
|
|
687 |
overridableMmpKeywords << QLatin1String(MMP_TARGETTYPE) << QLatin1String(MMP_OPTION_CW)
|
|
688 |
<< QLatin1String(MMP_OPTION_ARMCC) << QLatin1String(MMP_OPTION_GCCE);
|
|
689 |
|
|
690 |
foreach (QString item, project->values("MMP_RULES")) {
|
|
691 |
if (project->values(item).isEmpty()) {
|
|
692 |
checkOverridability(overridableMmpKeywords, item);
|
|
693 |
} else {
|
|
694 |
foreach (QString itemRow, project->values(item)) {
|
|
695 |
checkOverridability(overridableMmpKeywords, itemRow);
|
|
696 |
}
|
|
697 |
}
|
|
698 |
}
|
|
699 |
}
|
|
700 |
|
|
701 |
void SymbianMakefileGenerator::checkOverridability(QStringList &overridableKeywords, QString &checkString)
|
|
702 |
{
|
|
703 |
// Check if checkString contains overridable keyword and
|
|
704 |
// add the keyword to overridden keywords list if so.
|
|
705 |
QString simplifiedString = checkString.simplified();
|
|
706 |
foreach (QString item, overridableKeywords) {
|
|
707 |
if (simplifiedString.startsWith(item))
|
|
708 |
appendIfnotExist(overriddenMmpKeywords, item);
|
|
709 |
}
|
|
710 |
}
|
|
711 |
|
|
712 |
bool SymbianMakefileGenerator::removeDuplicatedStrings(QStringList &stringList)
|
|
713 |
{
|
|
714 |
QStringList tmpStringList;
|
|
715 |
|
|
716 |
for (int i = 0; i < stringList.size(); ++i) {
|
|
717 |
QString string = stringList.at(i);
|
|
718 |
if (tmpStringList.contains(string))
|
|
719 |
continue;
|
|
720 |
else
|
|
721 |
tmpStringList.append(string);
|
|
722 |
}
|
|
723 |
|
|
724 |
stringList.clear();
|
|
725 |
stringList = tmpStringList;
|
|
726 |
return true;
|
|
727 |
}
|
|
728 |
|
|
729 |
void SymbianMakefileGenerator::writeMmpFileHeader(QTextStream &t)
|
|
730 |
{
|
|
731 |
t << "// ==============================================================================" << endl;
|
|
732 |
t << "// Generated by qmake (" << qmake_version() << ") (Qt " << QT_VERSION_STR << ") on: ";
|
|
733 |
t << QDateTime::currentDateTime().toString(Qt::ISODate) << endl;
|
|
734 |
t << "// This file is generated by qmake and should not be modified by the" << endl;
|
|
735 |
t << "// user." << endl;
|
|
736 |
t << "// Name : " << escapeFilePath(fileFixify(project->projectFile().remove(project->projectFile().length() - 4, 4))) << Option::mmp_ext << endl;
|
|
737 |
t << "// ==============================================================================" << endl << endl;
|
|
738 |
}
|
|
739 |
|
|
740 |
void SymbianMakefileGenerator::writeMmpFile(QString &filename, QStringList &symbianLangCodes)
|
|
741 |
{
|
|
742 |
QFile ft(filename);
|
|
743 |
if (ft.open(QIODevice::WriteOnly)) {
|
|
744 |
if (Option::mkfile::listgen) {
|
|
745 |
generatePrint(fileInfo(ft.fileName()).absoluteFilePath());
|
|
746 |
}
|
|
747 |
|
|
748 |
generatedFiles << ft.fileName();
|
|
749 |
|
|
750 |
QTextStream t(&ft);
|
|
751 |
|
|
752 |
writeMmpFileHeader(t);
|
|
753 |
|
|
754 |
writeMmpFileTargetPart(t);
|
|
755 |
|
|
756 |
writeMmpFileResourcePart(t, symbianLangCodes);
|
|
757 |
|
|
758 |
writeMmpFileMacrosPart(t);
|
|
759 |
|
|
760 |
writeMmpFileIncludePart(t);
|
|
761 |
|
|
762 |
QDir current = QDir::current();
|
|
763 |
|
|
764 |
for (QMap<QString, QStringList>::iterator it = sources.begin(); it != sources.end(); ++it) {
|
|
765 |
QStringList values = it.value();
|
|
766 |
QString currentSourcePath = it.key();
|
|
767 |
|
|
768 |
if (values.size())
|
|
769 |
t << "SOURCEPATH \t" << fixPathForMmp(currentSourcePath, current) << endl;
|
|
770 |
|
|
771 |
for (int i = 0; i < values.size(); ++i) {
|
|
772 |
QString sourceFileName = values.at(i);
|
|
773 |
t << "SOURCE\t\t" << sourceFileName << endl;
|
|
774 |
}
|
|
775 |
t << endl;
|
|
776 |
}
|
|
777 |
t << endl;
|
|
778 |
|
|
779 |
if (!project->values("CONFIG").contains("static") && !project->values("CONFIG").contains("staticlib")) {
|
|
780 |
writeMmpFileLibraryPart(t);
|
|
781 |
}
|
|
782 |
|
|
783 |
writeMmpFileCapabilityPart(t);
|
|
784 |
|
|
785 |
writeMmpFileCompilerOptionPart(t);
|
|
786 |
|
|
787 |
writeMmpFileBinaryVersionPart(t);
|
|
788 |
|
|
789 |
writeMmpFileRulesPart(t);
|
|
790 |
} else {
|
|
791 |
PRINT_FILE_CREATE_ERROR(filename)
|
|
792 |
}
|
|
793 |
}
|
|
794 |
|
|
795 |
void SymbianMakefileGenerator::writeMmpFileMacrosPart(QTextStream& t)
|
|
796 |
{
|
|
797 |
t << endl;
|
|
798 |
|
|
799 |
QStringList &defines = project->values("DEFINES");
|
|
800 |
if (defines.size())
|
|
801 |
t << "// Qt Macros" << endl;
|
|
802 |
for (int i = 0; i < defines.size(); ++i) {
|
|
803 |
QString def = defines.at(i);
|
|
804 |
addMacro(t, def);
|
|
805 |
}
|
|
806 |
|
|
807 |
// These are required in order that all methods will be correctly exported e.g from qtestlib
|
|
808 |
QStringList &exp_defines = project->values("PRL_EXPORT_DEFINES");
|
|
809 |
if (exp_defines.size())
|
|
810 |
t << endl << "// Qt Export Defines" << endl;
|
|
811 |
for (int i = 0; i < exp_defines.size(); ++i) {
|
|
812 |
QString def = exp_defines.at(i);
|
|
813 |
addMacro(t, def);
|
|
814 |
}
|
|
815 |
|
|
816 |
t << endl;
|
|
817 |
}
|
|
818 |
|
|
819 |
void SymbianMakefileGenerator::addMacro(QTextStream& t, const QString& value)
|
|
820 |
{
|
|
821 |
t << "MACRO" << "\t\t" << value << endl;
|
|
822 |
}
|
|
823 |
|
|
824 |
|
|
825 |
void SymbianMakefileGenerator::writeMmpFileTargetPart(QTextStream& t)
|
|
826 |
{
|
|
827 |
bool skipTargetType = overriddenMmpKeywords.contains(MMP_TARGETTYPE);
|
|
828 |
|
|
829 |
if (targetType == TypeExe) {
|
|
830 |
t << MMP_TARGET << "\t\t" << fixedTarget << ".exe" << endl;
|
|
831 |
if (!skipTargetType) {
|
|
832 |
if (project->values("CONFIG").contains("stdbinary", Qt::CaseInsensitive))
|
|
833 |
t << MMP_TARGETTYPE << "\t\t" << "STDEXE" << endl;
|
|
834 |
else
|
|
835 |
t << MMP_TARGETTYPE << "\t\t" << "EXE" << endl;
|
|
836 |
}
|
|
837 |
} else if (targetType == TypeDll || targetType == TypePlugin) {
|
|
838 |
t << MMP_TARGET << "\t\t" << fixedTarget << ".dll" << endl;
|
|
839 |
if (!skipTargetType) {
|
|
840 |
if (project->values("CONFIG").contains("stdbinary", Qt::CaseInsensitive))
|
|
841 |
t << MMP_TARGETTYPE << "\t\t" << "STDDLL" << endl;
|
|
842 |
else
|
|
843 |
t << MMP_TARGETTYPE << "\t\t" << "DLL" << endl;
|
|
844 |
}
|
|
845 |
} else if (targetType == TypeLib) {
|
|
846 |
t << MMP_TARGET << "\t\t" << fixedTarget << ".lib" << endl;
|
|
847 |
if (!skipTargetType) {
|
|
848 |
if (project->values("CONFIG").contains("stdbinary", Qt::CaseInsensitive))
|
|
849 |
t << MMP_TARGETTYPE << "\t\t" << "STDLIB" << endl;
|
|
850 |
else
|
|
851 |
t << MMP_TARGETTYPE << "\t\t" << "LIB" << endl;
|
|
852 |
}
|
|
853 |
} else {
|
|
854 |
fprintf(stderr, "Error: Unexpected targettype (%d) in SymbianMakefileGenerator::writeMmpFileTargetPart\n", targetType);
|
|
855 |
}
|
|
856 |
|
|
857 |
t << endl;
|
|
858 |
|
|
859 |
t << "UID" << "\t\t" << uid2 << " " << uid3 << endl;
|
|
860 |
|
|
861 |
if (0 != project->values("TARGET.SID").size()) {
|
|
862 |
t << MMP_SECUREID << "\t\t" << project->values("TARGET.SID").join(" ") << endl;
|
|
863 |
} else {
|
|
864 |
if (0 == uid3.size())
|
|
865 |
t << MMP_SECUREID << "\t\t" << "0" << endl;
|
|
866 |
else
|
|
867 |
t << MMP_SECUREID << "\t\t" << uid3 << endl;
|
|
868 |
}
|
|
869 |
|
|
870 |
// default value used from mkspecs is 0
|
|
871 |
if (0 != project->values("TARGET.VID").size()) {
|
|
872 |
t << "VENDORID" << "\t\t" << project->values("TARGET.VID").join(" ") << endl;
|
|
873 |
}
|
|
874 |
|
|
875 |
t << endl;
|
|
876 |
|
|
877 |
if (0 != project->first("TARGET.EPOCSTACKSIZE").size())
|
|
878 |
t << "EPOCSTACKSIZE" << "\t\t" << project->first("TARGET.EPOCSTACKSIZE") << endl;
|
|
879 |
if (0 != project->values("TARGET.EPOCHEAPSIZE").size())
|
|
880 |
t << "EPOCHEAPSIZE" << "\t\t" << project->values("TARGET.EPOCHEAPSIZE").join(" ") << endl;
|
|
881 |
if (0 != project->values("TARGET.EPOCALLOWDLLDATA").size())
|
|
882 |
t << "EPOCALLOWDLLDATA" << endl;
|
|
883 |
|
|
884 |
if (targetType == TypePlugin && !project->values("CONFIG").contains("stdbinary", Qt::CaseInsensitive)) {
|
|
885 |
// Use custom def file for Qt plugins
|
|
886 |
t << "DEFFILE " PLUGIN_COMMON_DEF_FILE_FOR_MMP << endl;
|
|
887 |
}
|
|
888 |
|
|
889 |
t << endl;
|
|
890 |
}
|
|
891 |
|
|
892 |
|
|
893 |
/*
|
|
894 |
Application registration resource files should be installed to the
|
|
895 |
\private\10003a3f\import\apps directory.
|
|
896 |
*/
|
|
897 |
void SymbianMakefileGenerator::writeMmpFileResourcePart(QTextStream& t, QStringList &symbianLangCodes)
|
|
898 |
{
|
|
899 |
if ((targetType == TypeExe) &&
|
|
900 |
!project->values("CONFIG").contains("no_icon", Qt::CaseInsensitive)) {
|
|
901 |
|
|
902 |
QString locTarget = fixedTarget;
|
|
903 |
locTarget.append(".rss");
|
|
904 |
|
|
905 |
t << "SOURCEPATH\t\t\t. " << endl;
|
|
906 |
t << "LANG SC "; // no endl
|
|
907 |
foreach(QString lang, symbianLangCodes) {
|
|
908 |
t << lang << " "; // no endl
|
|
909 |
}
|
|
910 |
t << endl;
|
|
911 |
t << "START RESOURCE\t\t" << locTarget << endl;
|
|
912 |
t << "HEADER" << endl;
|
|
913 |
t << "TARGETPATH\t\t\t" RESOURCE_DIRECTORY_MMP << endl;
|
|
914 |
t << "END" << endl << endl;
|
|
915 |
|
|
916 |
QString regTarget = fixedTarget;
|
|
917 |
regTarget.append("_reg.rss");
|
|
918 |
|
|
919 |
t << "SOURCEPATH\t\t\t." << endl;
|
|
920 |
t << "START RESOURCE\t\t" << regTarget << endl;
|
|
921 |
if (isForSymbianSbsv2())
|
|
922 |
t << "DEPENDS " << fixedTarget << ".rsg" << endl;
|
|
923 |
t << "TARGETPATH\t\t" REGISTRATION_RESOURCE_DIRECTORY_HW << endl;
|
|
924 |
t << "END" << endl << endl;
|
|
925 |
}
|
|
926 |
}
|
|
927 |
|
|
928 |
void SymbianMakefileGenerator::writeMmpFileSystemIncludePart(QTextStream& t)
|
|
929 |
{
|
|
930 |
QDir current = QDir::current();
|
|
931 |
|
|
932 |
for (QMap<QString, QStringList>::iterator it = systeminclude.begin(); it != systeminclude.end(); ++it) {
|
|
933 |
QStringList values = it.value();
|
|
934 |
for (int i = 0; i < values.size(); ++i) {
|
|
935 |
QString handledPath = values.at(i);
|
|
936 |
t << "SYSTEMINCLUDE" << "\t\t" << fixPathForMmp(handledPath, current) << endl;
|
|
937 |
}
|
|
938 |
}
|
|
939 |
|
|
940 |
t << endl;
|
|
941 |
}
|
|
942 |
|
|
943 |
void SymbianMakefileGenerator::writeMmpFileIncludePart(QTextStream& t)
|
|
944 |
{
|
|
945 |
writeMmpFileSystemIncludePart(t);
|
|
946 |
}
|
|
947 |
|
|
948 |
void SymbianMakefileGenerator::writeMmpFileLibraryPart(QTextStream& t)
|
|
949 |
{
|
|
950 |
QStringList &libs = project->values("LIBS");
|
|
951 |
libs << project->values("QMAKE_LIBS") << project->values("QMAKE_LIBS_PRIVATE");
|
|
952 |
|
|
953 |
removeDuplicatedStrings(libs);
|
|
954 |
|
|
955 |
for (int i = 0; i < libs.size(); ++i) {
|
|
956 |
QString lib = libs.at(i);
|
|
957 |
// The -L flag is uninteresting, since all symbian libraries exist in the same directory.
|
|
958 |
if (lib.startsWith("-l")) {
|
|
959 |
lib.remove(0, 2);
|
|
960 |
QString mmpStatement;
|
|
961 |
if (lib.endsWith(".dll")) {
|
|
962 |
lib.chop(4);
|
|
963 |
mmpStatement = "LIBRARY\t\t";
|
|
964 |
} else if (lib.endsWith(".lib")) {
|
|
965 |
lib.chop(4);
|
|
966 |
mmpStatement = "STATICLIBRARY\t";
|
|
967 |
} else {
|
|
968 |
// Hacky way to find out what kind of library it is. Check the
|
|
969 |
// ARMV5 build directory for library type. We default to shared
|
|
970 |
// library, since that is more common.
|
|
971 |
QString udebStaticLibLocation(epocRoot());
|
|
972 |
QString urelStaticLibLocation(udebStaticLibLocation);
|
|
973 |
udebStaticLibLocation += QString("epoc32/release/armv5/udeb/%1.lib").arg(lib);
|
|
974 |
urelStaticLibLocation += QString("epoc32/release/armv5/urel/%1.lib").arg(lib);
|
|
975 |
if (QFile::exists(udebStaticLibLocation) || QFile::exists(urelStaticLibLocation)) {
|
|
976 |
mmpStatement = "STATICLIBRARY\t";
|
|
977 |
} else {
|
|
978 |
mmpStatement = "LIBRARY\t\t";
|
|
979 |
}
|
|
980 |
}
|
|
981 |
t << mmpStatement << lib << ".lib" << endl;
|
|
982 |
}
|
|
983 |
}
|
|
984 |
|
|
985 |
t << endl;
|
|
986 |
}
|
|
987 |
|
|
988 |
void SymbianMakefileGenerator::writeMmpFileCapabilityPart(QTextStream& t)
|
|
989 |
{
|
|
990 |
if (0 != project->first("TARGET.CAPABILITY").size()) {
|
|
991 |
QStringList &capabilities = project->values("TARGET.CAPABILITY");
|
|
992 |
t << "CAPABILITY" << "\t\t";
|
|
993 |
|
|
994 |
for (int i = 0; i < capabilities.size(); ++i) {
|
|
995 |
QString cap = capabilities.at(i);
|
|
996 |
t << cap << " ";
|
|
997 |
}
|
|
998 |
} else {
|
|
999 |
t << "CAPABILITY" << "\t\t" << "None";
|
|
1000 |
}
|
|
1001 |
t << endl << endl;
|
|
1002 |
}
|
|
1003 |
|
|
1004 |
void SymbianMakefileGenerator::writeMmpFileCompilerOptionPart(QTextStream& t)
|
|
1005 |
{
|
|
1006 |
QString cw, armcc, gcce;
|
|
1007 |
|
|
1008 |
if (0 != project->values("QMAKE_CXXFLAGS.CW").size()) {
|
|
1009 |
cw.append(project->values("QMAKE_CXXFLAGS.CW").join(" "));
|
|
1010 |
cw.append(" ");
|
|
1011 |
}
|
|
1012 |
|
|
1013 |
if (0 != project->values("QMAKE_CXXFLAGS.ARMCC").size()) {
|
|
1014 |
armcc.append(project->values("QMAKE_CXXFLAGS.ARMCC").join(" "));
|
|
1015 |
armcc.append(" ");
|
|
1016 |
}
|
|
1017 |
|
|
1018 |
if (0 != project->values("QMAKE_CXXFLAGS.GCCE").size()) {
|
|
1019 |
gcce.append(project->values("QMAKE_CXXFLAGS.GCCE").join(" "));
|
|
1020 |
gcce.append(" ");
|
|
1021 |
}
|
|
1022 |
|
|
1023 |
if (0 != project->values("QMAKE_CFLAGS.CW").size()) {
|
|
1024 |
cw.append(project->values("QMAKE_CFLAGS.CW").join(" "));
|
|
1025 |
cw.append(" ");
|
|
1026 |
}
|
|
1027 |
|
|
1028 |
if (0 != project->values("QMAKE_CFLAGS.ARMCC").size()) {
|
|
1029 |
armcc.append(project->values("QMAKE_CFLAGS.ARMCC").join(" "));
|
|
1030 |
armcc.append(" ");
|
|
1031 |
}
|
|
1032 |
|
|
1033 |
if (0 != project->values("QMAKE_CFLAGS.GCCE").size()) {
|
|
1034 |
gcce.append(project->values("QMAKE_CXXFLAGS.GCCE").join(" "));
|
|
1035 |
gcce.append(" ");
|
|
1036 |
}
|
|
1037 |
|
|
1038 |
if (0 != project->values("QMAKE_CXXFLAGS").size()) {
|
|
1039 |
cw.append(project->values("QMAKE_CXXFLAGS").join(" "));
|
|
1040 |
cw.append(" ");
|
|
1041 |
armcc.append(project->values("QMAKE_CXXFLAGS").join(" "));
|
|
1042 |
armcc.append(" ");
|
|
1043 |
gcce.append(project->values("QMAKE_CXXFLAGS").join(" "));
|
|
1044 |
gcce.append(" ");
|
|
1045 |
}
|
|
1046 |
|
|
1047 |
if (0 != project->values("QMAKE_CFLAGS").size()) {
|
|
1048 |
cw.append(project->values("QMAKE_CFLAGS").join(" "));
|
|
1049 |
cw.append(" ");
|
|
1050 |
armcc.append(project->values("QMAKE_CFLAGS").join(" "));
|
|
1051 |
armcc.append(" ");
|
|
1052 |
gcce.append(project->values("QMAKE_CFLAGS").join(" "));
|
|
1053 |
gcce.append(" ");
|
|
1054 |
}
|
|
1055 |
|
|
1056 |
if (!cw.isEmpty() && cw[cw.size()-1] == ' ')
|
|
1057 |
cw.chop(1);
|
|
1058 |
if (!armcc.isEmpty() && armcc[armcc.size()-1] == ' ')
|
|
1059 |
armcc.chop(1);
|
|
1060 |
if (!gcce.isEmpty() && gcce[gcce.size()-1] == ' ')
|
|
1061 |
gcce.chop(1);
|
|
1062 |
|
|
1063 |
if (!cw.isEmpty() && !overriddenMmpKeywords.contains(MMP_OPTION_CW))
|
|
1064 |
t << MMP_OPTION_CW " " << cw << endl;
|
|
1065 |
if (!armcc.isEmpty() && !overriddenMmpKeywords.contains(MMP_OPTION_ARMCC))
|
|
1066 |
t << MMP_OPTION_ARMCC " " << armcc << endl;
|
|
1067 |
if (!gcce.isEmpty() && !overriddenMmpKeywords.contains(MMP_OPTION_GCCE))
|
|
1068 |
t << MMP_OPTION_GCCE " " << gcce << endl;
|
|
1069 |
|
|
1070 |
t << endl;
|
|
1071 |
}
|
|
1072 |
|
|
1073 |
void SymbianMakefileGenerator::writeMmpFileBinaryVersionPart(QTextStream& t)
|
|
1074 |
{
|
|
1075 |
QString applicationVersion = project->first("VERSION");
|
|
1076 |
QStringList verNumList = applicationVersion.split('.');
|
|
1077 |
uint major = 0;
|
|
1078 |
uint minor = 0;
|
|
1079 |
uint patch = 0;
|
|
1080 |
bool success = false;
|
|
1081 |
|
|
1082 |
if (verNumList.size() > 0) {
|
|
1083 |
major = verNumList[0].toUInt(&success);
|
|
1084 |
if (success && verNumList.size() > 1) {
|
|
1085 |
minor = verNumList[1].toUInt(&success);
|
|
1086 |
if (success && verNumList.size() > 2) {
|
|
1087 |
patch = verNumList[2].toUInt(&success);
|
|
1088 |
}
|
|
1089 |
}
|
|
1090 |
}
|
|
1091 |
|
|
1092 |
QString mmpVersion;
|
|
1093 |
if (success && major <= 0xFFFF && minor <= 0xFF && patch <= 0xFF) {
|
|
1094 |
// Symbian binary version only has major and minor components, so compress
|
|
1095 |
// Qt's minor and patch values into the minor component. Since Symbian's minor
|
|
1096 |
// component is a 16 bit value, only allow 8 bits for each to avoid overflow.
|
|
1097 |
mmpVersion.append(QString::number(major))
|
|
1098 |
.append('.')
|
|
1099 |
.append(QString::number((minor << 8) + patch));
|
|
1100 |
} else {
|
|
1101 |
if (!applicationVersion.isEmpty())
|
|
1102 |
fprintf(stderr, "Invalid VERSION string: %s\n", qPrintable(applicationVersion));
|
|
1103 |
mmpVersion = "10.0"; // Default binary version for symbian is 10.0
|
|
1104 |
}
|
|
1105 |
|
|
1106 |
t << "VERSION " << mmpVersion << endl;
|
|
1107 |
}
|
|
1108 |
|
|
1109 |
void SymbianMakefileGenerator::writeMmpFileRulesPart(QTextStream& t)
|
|
1110 |
{
|
|
1111 |
foreach(QString item, project->values("MMP_RULES")) {
|
|
1112 |
t << endl;
|
|
1113 |
// If there is no stringlist defined for a rule, use rule name directly
|
|
1114 |
// This is convenience for defining single line mmp statements
|
|
1115 |
if (project->values(item).isEmpty()) {
|
|
1116 |
t << item << endl;
|
|
1117 |
} else {
|
|
1118 |
foreach(QString itemRow, project->values(item)) {
|
|
1119 |
t << itemRow << endl;
|
|
1120 |
}
|
|
1121 |
}
|
|
1122 |
}
|
|
1123 |
}
|
|
1124 |
|
|
1125 |
void SymbianMakefileGenerator::writeBldInfContent(QTextStream &t, bool addDeploymentExtension)
|
|
1126 |
{
|
|
1127 |
// Read user defined bld inf rules
|
|
1128 |
|
|
1129 |
QMap<QString, QStringList> userBldInfRules;
|
|
1130 |
for (QMap<QString, QStringList>::iterator it = project->variables().begin(); it != project->variables().end(); ++it) {
|
|
1131 |
if (it.key().startsWith(BLD_INF_RULES_BASE)) {
|
|
1132 |
QString newKey = it.key().mid(sizeof(BLD_INF_RULES_BASE) - 1);
|
|
1133 |
if (newKey.isEmpty()) {
|
|
1134 |
fprintf(stderr, "Warning: Empty BLD_INF_RULES key encountered\n");
|
|
1135 |
continue;
|
|
1136 |
}
|
|
1137 |
QStringList newValues;
|
|
1138 |
QStringList values = it.value();
|
|
1139 |
foreach(QString item, values) {
|
|
1140 |
// If there is no stringlist defined for a rule, use rule name directly
|
|
1141 |
// This is convenience for defining single line statements
|
|
1142 |
if (project->values(item).isEmpty()) {
|
|
1143 |
newValues << item;
|
|
1144 |
} else {
|
|
1145 |
foreach(QString itemRow, project->values(item)) {
|
|
1146 |
newValues << itemRow;
|
|
1147 |
}
|
|
1148 |
}
|
|
1149 |
}
|
|
1150 |
userBldInfRules.insert(newKey, newValues);
|
|
1151 |
}
|
|
1152 |
}
|
|
1153 |
|
|
1154 |
// Add includes of subdirs bld.inf files
|
|
1155 |
|
|
1156 |
QString mmpfilename = escapeFilePath(fileFixify(project->projectFile()));
|
|
1157 |
mmpfilename = mmpfilename.replace(mmpfilename.lastIndexOf("."), 4, Option::mmp_ext);
|
|
1158 |
QString currentPath = qmake_getpwd();
|
|
1159 |
QDir directory(currentPath);
|
|
1160 |
|
|
1161 |
const QStringList &subdirs = project->values("SUBDIRS");
|
|
1162 |
foreach(QString item, subdirs) {
|
|
1163 |
QString fixedItem;
|
|
1164 |
if (!project->isEmpty(item + ".file")) {
|
|
1165 |
fixedItem = project->first(item + ".file");
|
|
1166 |
} else if (!project->isEmpty(item + ".subdir")) {
|
|
1167 |
fixedItem = project->first(item + ".subdir");
|
|
1168 |
} else {
|
|
1169 |
fixedItem = item;
|
|
1170 |
}
|
|
1171 |
|
|
1172 |
QFileInfo subdir(fileInfo(fixedItem));
|
|
1173 |
QString relativePath = directory.relativeFilePath(fixedItem);
|
|
1174 |
QString subdirFileName = subdir.completeBaseName();
|
|
1175 |
QString fullProName = subdir.absoluteFilePath();;
|
|
1176 |
QString bldinfFilename;
|
|
1177 |
|
|
1178 |
if (subdir.isDir()) {
|
|
1179 |
// Subdir is a regular project
|
|
1180 |
bldinfFilename = relativePath + QString("/") + QString(BLD_INF_FILENAME);
|
|
1181 |
fullProName += QString("/") + subdirFileName + Option::pro_ext;
|
|
1182 |
} else {
|
|
1183 |
// Subdir is actually a .pro file
|
|
1184 |
if (relativePath.contains("/")) {
|
|
1185 |
// .pro not in same directory as parent .pro
|
|
1186 |
relativePath.remove(relativePath.lastIndexOf("/") + 1, relativePath.length());
|
|
1187 |
bldinfFilename = relativePath;
|
|
1188 |
} else {
|
|
1189 |
// .pro and parent .pro in same directory
|
|
1190 |
bldinfFilename = QString("./");
|
|
1191 |
}
|
|
1192 |
bldinfFilename += QString(BLD_INF_FILENAME ".") + subdirFileName;
|
|
1193 |
}
|
|
1194 |
|
|
1195 |
QString uid = generate_uid(fullProName);
|
|
1196 |
QString bldinfDefine = QString("BLD_INF_") + subdirFileName + QString("_") + uid;
|
|
1197 |
bldinfDefine = bldinfDefine.toUpper();
|
|
1198 |
removeSpecialCharacters(bldinfDefine);
|
|
1199 |
|
|
1200 |
t << "#ifndef " << bldinfDefine << endl;
|
|
1201 |
t << "\t#include \"" << QDir::toNativeSeparators(bldinfFilename) << "\"" << endl;
|
|
1202 |
t << "#endif // " << bldinfDefine << endl;
|
|
1203 |
}
|
|
1204 |
|
|
1205 |
// Add supported project platforms
|
|
1206 |
|
|
1207 |
t << endl << BLD_INF_TAG_PLATFORMS << endl << endl;
|
|
1208 |
if (0 != project->values("SYMBIAN_PLATFORMS").size())
|
|
1209 |
t << project->values("SYMBIAN_PLATFORMS").join(" ") << endl;
|
|
1210 |
|
|
1211 |
QStringList userItems = userBldInfRules.value(BLD_INF_TAG_PLATFORMS);
|
|
1212 |
foreach(QString item, userItems)
|
|
1213 |
t << item << endl;
|
|
1214 |
userBldInfRules.remove(BLD_INF_TAG_PLATFORMS);
|
|
1215 |
t << endl;
|
|
1216 |
|
|
1217 |
// Add project mmps and old style extension makefiles
|
|
1218 |
|
|
1219 |
QString mmpTag;
|
|
1220 |
if (project->values("CONFIG").contains("symbian_test", Qt::CaseInsensitive))
|
|
1221 |
mmpTag = QLatin1String(BLD_INF_TAG_TESTMMPFILES);
|
|
1222 |
else
|
|
1223 |
mmpTag = QLatin1String(BLD_INF_TAG_MMPFILES);
|
|
1224 |
|
|
1225 |
t << endl << mmpTag << endl << endl;
|
|
1226 |
|
|
1227 |
writeBldInfMkFilePart(t, addDeploymentExtension);
|
|
1228 |
if (targetType != TypeSubdirs) {
|
|
1229 |
QString shortProFilename = project->projectFile();
|
|
1230 |
shortProFilename.replace(0, shortProFilename.lastIndexOf("/") + 1, QString(""));
|
|
1231 |
shortProFilename.replace(Option::pro_ext, QString(""));
|
|
1232 |
|
|
1233 |
QString mmpFilename = shortProFilename + QString("_") + uid3 + Option::mmp_ext;
|
|
1234 |
|
|
1235 |
t << mmpFilename << endl;
|
|
1236 |
}
|
|
1237 |
|
|
1238 |
userItems = userBldInfRules.value(mmpTag);
|
|
1239 |
foreach(QString item, userItems)
|
|
1240 |
t << item << endl;
|
|
1241 |
userBldInfRules.remove(mmpTag);
|
|
1242 |
|
|
1243 |
t << endl << BLD_INF_TAG_EXTENSIONS << endl << endl;
|
|
1244 |
|
|
1245 |
// Generate extension rules
|
|
1246 |
|
|
1247 |
writeBldInfExtensionRulesPart(t);
|
|
1248 |
|
|
1249 |
userItems = userBldInfRules.value(BLD_INF_TAG_EXTENSIONS);
|
|
1250 |
foreach(QString item, userItems)
|
|
1251 |
t << item << endl;
|
|
1252 |
userBldInfRules.remove(BLD_INF_TAG_EXTENSIONS);
|
|
1253 |
|
|
1254 |
// Add rest of the user defined content
|
|
1255 |
|
|
1256 |
for (QMap<QString, QStringList>::iterator it = userBldInfRules.begin(); it != userBldInfRules.end(); ++it) {
|
|
1257 |
t << endl << endl << it.key() << endl << endl;
|
|
1258 |
userItems = it.value();
|
|
1259 |
foreach(QString item, userItems)
|
|
1260 |
t << item << endl;
|
|
1261 |
}
|
|
1262 |
if (project->values("CONFIG").contains("headerexport", Qt::CaseInsensitive)) {
|
|
1263 |
writeExportPart(t);
|
|
1264 |
}
|
|
1265 |
}
|
|
1266 |
|
|
1267 |
void SymbianMakefileGenerator::writeExportPart(QTextStream &t)
|
|
1268 |
{
|
|
1269 |
QDir currentDir = QDir::current();
|
|
1270 |
t << "prj_exports" << endl;
|
|
1271 |
|
|
1272 |
foreach(QString install, project->values("INSTALLS")) {
|
|
1273 |
QString installDir = project->first(install + ".path");
|
|
1274 |
|
|
1275 |
// Export macros are the recommended way, but might not be used
|
|
1276 |
// in all cases (such as Qt headers). We handle it either way,
|
|
1277 |
// macros or not.
|
|
1278 |
bool useLayerMacro = false;
|
|
1279 |
if (installDir.startsWith("MW_LAYER_") ||
|
|
1280 |
installDir.startsWith("APP_LAYER_") ||
|
|
1281 |
installDir.startsWith("APP_LAYER_")) {
|
|
1282 |
useLayerMacro = true;
|
|
1283 |
} else {
|
|
1284 |
#ifdef Q_OS_WIN
|
|
1285 |
// If we are running on Windows, and the export target starts
|
|
1286 |
// with a drive letter, we need to remove it, the build tools
|
|
1287 |
// cannot handle drives in export paths
|
|
1288 |
|
|
1289 |
if (installDir.length() > 2 && installDir[1] == ':') {
|
|
1290 |
installDir.remove(0, 2);
|
|
1291 |
}
|
|
1292 |
#endif
|
|
1293 |
installDir.replace("\\", "/");
|
|
1294 |
if (!installDir.endsWith("/")) {
|
|
1295 |
installDir.append("/");
|
|
1296 |
}
|
|
1297 |
}
|
|
1298 |
|
|
1299 |
foreach(QString target, project->values(install + ".files")) {
|
|
1300 |
// Handle glob targets (only in the current directory)
|
|
1301 |
if (target.contains("*") || target.contains("?")) {
|
|
1302 |
QDir globber = QDir::current();
|
|
1303 |
globber.setNameFilters(QStringList(target));
|
|
1304 |
foreach(QString globbedTarget, globber.entryList()) {
|
|
1305 |
if (useLayerMacro) {
|
|
1306 |
QFileInfo globbedInfo(globbedTarget);
|
|
1307 |
t << globbedTarget << " " << installDir <<
|
|
1308 |
"(" << globbedInfo.fileName() << ")" << endl;
|
|
1309 |
} else {
|
|
1310 |
t << globbedTarget << " " << installDir <<
|
|
1311 |
globbedTarget << endl;
|
|
1312 |
}
|
|
1313 |
}
|
|
1314 |
} else {
|
|
1315 |
target = currentDir.relativeFilePath(target);
|
|
1316 |
QFileInfo targetInfo(target);
|
|
1317 |
|
|
1318 |
// Not possible to export whole directories, extension makefile needed for this
|
|
1319 |
// Note that this check requires the file to be exported to exist in the qmake phase!
|
|
1320 |
if (targetInfo.isFile()) {
|
|
1321 |
if (useLayerMacro) {
|
|
1322 |
t << target << " " << installDir <<
|
|
1323 |
"(" << targetInfo.fileName() << ")" << endl;
|
|
1324 |
} else {
|
|
1325 |
t << target << " " << installDir <<
|
|
1326 |
targetInfo.fileName() << endl;
|
|
1327 |
}
|
|
1328 |
}
|
|
1329 |
}
|
|
1330 |
}
|
|
1331 |
}
|
|
1332 |
t << endl;
|
|
1333 |
}
|
|
1334 |
|
|
1335 |
void SymbianMakefileGenerator::writeRegRssFile(QStringList &userItems)
|
|
1336 |
{
|
|
1337 |
QString filename(fixedTarget);
|
|
1338 |
filename.append("_reg.rss");
|
|
1339 |
QFile ft(filename);
|
|
1340 |
if (ft.open(QIODevice::WriteOnly)) {
|
|
1341 |
if (Option::mkfile::listgen) {
|
|
1342 |
generatePrint(fileInfo(ft.fileName()).absoluteFilePath());
|
|
1343 |
}
|
|
1344 |
generatedFiles << ft.fileName();
|
|
1345 |
QTextStream t(&ft);
|
|
1346 |
t << "// ============================================================================" << endl;
|
|
1347 |
t << "// * Generated by qmake (" << qmake_version() << ") (Qt " << QT_VERSION_STR << ") on: ";
|
|
1348 |
t << QDateTime::currentDateTime().toString(Qt::ISODate) << endl;
|
|
1349 |
t << "// * This file is generated by qmake and should not be modified by the" << endl;
|
|
1350 |
t << "// * user." << endl;
|
|
1351 |
t << "// ============================================================================" << endl;
|
|
1352 |
t << endl;
|
|
1353 |
t << "#include <" << fixedTarget << ".rsg>" << endl;
|
|
1354 |
t << "#include <appinfo.rh>" << endl;
|
|
1355 |
t << endl;
|
|
1356 |
//t << "#include <data_caging_paths.hrh>" << "\n" << endl;
|
|
1357 |
t << "UID2 " << "KUidAppRegistrationResourceFile" << endl;
|
|
1358 |
t << "UID3 " << uid3 << endl << endl;
|
|
1359 |
t << "RESOURCE APP_REGISTRATION_INFO" << endl;
|
|
1360 |
t << "\t{" << endl;
|
|
1361 |
t << "\tapp_file=\"" << fixedTarget << "\";" << endl;
|
|
1362 |
t << "\tlocalisable_resource_file=\"" RESOURCE_DIRECTORY_RESOURCE << fixedTarget << "\";" << endl;
|
|
1363 |
t << endl;
|
|
1364 |
|
|
1365 |
foreach(QString item, userItems)
|
|
1366 |
t << "\t" << item << endl;
|
|
1367 |
t << "\t}" << endl;
|
|
1368 |
} else {
|
|
1369 |
PRINT_FILE_CREATE_ERROR(filename)
|
|
1370 |
}
|
|
1371 |
}
|
|
1372 |
|
|
1373 |
void SymbianMakefileGenerator::writeRssFile(QString &numberOfIcons, QString &iconFile)
|
|
1374 |
{
|
|
1375 |
QString filename(fixedTarget);
|
|
1376 |
filename.append(".rss");
|
|
1377 |
QFile ft(filename);
|
|
1378 |
if (ft.open(QIODevice::WriteOnly)) {
|
|
1379 |
if (Option::mkfile::listgen) {
|
|
1380 |
generatePrint(fileInfo(ft.fileName()).absoluteFilePath());
|
|
1381 |
}
|
|
1382 |
generatedFiles << ft.fileName();
|
|
1383 |
QTextStream t(&ft);
|
|
1384 |
t << "// ============================================================================" << endl;
|
|
1385 |
t << "// * Generated by qmake (" << qmake_version() << ") (Qt " << QT_VERSION_STR << ") on: ";
|
|
1386 |
t << QDateTime::currentDateTime().toString(Qt::ISODate) << endl;
|
|
1387 |
t << "// * This file is generated by qmake and should not be modified by the" << endl;
|
|
1388 |
t << "// * user." << endl;
|
|
1389 |
t << "// ============================================================================" << endl;
|
|
1390 |
t << endl;
|
|
1391 |
t << "#include <appinfo.rh>" << endl;
|
|
1392 |
t << "#include \"" << fixedTarget << ".loc\"" << endl;
|
|
1393 |
t << endl;
|
|
1394 |
t << "RESOURCE LOCALISABLE_APP_INFO r_localisable_app_info" << endl;
|
|
1395 |
t << "\t{" << endl;
|
|
1396 |
t << "\tshort_caption = STRING_r_short_caption;" << endl;
|
|
1397 |
t << "\tcaption_and_icon =" << endl;
|
|
1398 |
t << "\tCAPTION_AND_ICON_INFO" << endl;
|
|
1399 |
t << "\t\t{" << endl;
|
|
1400 |
t << "\t\tcaption = STRING_r_caption;" << endl;
|
|
1401 |
|
|
1402 |
if (numberOfIcons.isEmpty() || iconFile.isEmpty()) {
|
|
1403 |
// There can be maximum one item in this tag, validated when parsed
|
|
1404 |
t << "\t\tnumber_of_icons = 0;" << endl;
|
|
1405 |
t << "\t\ticon_file = \"\";" << endl;
|
|
1406 |
} else {
|
|
1407 |
// There can be maximum one item in this tag, validated when parsed
|
|
1408 |
t << "\t\tnumber_of_icons = " << numberOfIcons << ";" << endl;
|
|
1409 |
t << "\t\ticon_file = \"" << iconFile << "\";" << endl;
|
|
1410 |
}
|
|
1411 |
t << "\t\t};" << endl;
|
|
1412 |
t << "\t}" << endl;
|
|
1413 |
t << endl;
|
|
1414 |
} else {
|
|
1415 |
PRINT_FILE_CREATE_ERROR(filename);
|
|
1416 |
}
|
|
1417 |
}
|
|
1418 |
|
|
1419 |
void SymbianMakefileGenerator::writeLocFile(QStringList &symbianLangCodes)
|
|
1420 |
{
|
|
1421 |
QString filename(fixedTarget);
|
|
1422 |
filename.append(".loc");
|
|
1423 |
QFile ft(filename);
|
|
1424 |
if (ft.open(QIODevice::WriteOnly)) {
|
|
1425 |
if (Option::mkfile::listgen) {
|
|
1426 |
generatePrint(fileInfo(ft.fileName()).absoluteFilePath());
|
|
1427 |
}
|
|
1428 |
generatedFiles << ft.fileName();
|
|
1429 |
QTextStream t(&ft);
|
|
1430 |
t << "// ============================================================================" << endl;
|
|
1431 |
t << "// * Generated by qmake (" << qmake_version() << ") (Qt " << QT_VERSION_STR << ") on: ";
|
|
1432 |
t << QDateTime::currentDateTime().toString(Qt::ISODate) << endl;
|
|
1433 |
t << "// * This file is generated by qmake and should not be modified by the" << endl;
|
|
1434 |
t << "// * user." << endl;
|
|
1435 |
t << "// ============================================================================" << endl;
|
|
1436 |
t << endl;
|
|
1437 |
t << "#ifdef LANGUAGE_SC" << endl;
|
|
1438 |
t << "#define STRING_r_short_caption \"" << fixedTarget << "\"" << endl;
|
|
1439 |
t << "#define STRING_r_caption \"" << fixedTarget << "\"" << endl;
|
|
1440 |
foreach(QString lang, symbianLangCodes) {
|
|
1441 |
t << "#elif defined LANGUAGE_" << lang << endl;
|
|
1442 |
t << "#define STRING_r_short_caption \"" << fixedTarget << "\"" << endl;
|
|
1443 |
t << "#define STRING_r_caption \"" << fixedTarget << "\"" << endl;
|
|
1444 |
}
|
|
1445 |
t << "#else" << endl;
|
|
1446 |
t << "#define STRING_r_short_caption \"" << fixedTarget << "\"" << endl;
|
|
1447 |
t << "#define STRING_r_caption \"" << fixedTarget << "\"" << endl;
|
|
1448 |
t << "#endif" << endl;
|
|
1449 |
} else {
|
|
1450 |
PRINT_FILE_CREATE_ERROR(filename);
|
|
1451 |
}
|
|
1452 |
}
|
|
1453 |
|
|
1454 |
void SymbianMakefileGenerator::readRssRules(QString &numberOfIcons, QString &iconFile, QStringList &userRssRules)
|
|
1455 |
{
|
|
1456 |
for (QMap<QString, QStringList>::iterator it = project->variables().begin(); it != project->variables().end(); ++it) {
|
|
1457 |
if (it.key().startsWith(RSS_RULES_BASE)) {
|
|
1458 |
QString newKey = it.key().mid(sizeof(RSS_RULES_BASE) - 1);
|
|
1459 |
if (newKey.isEmpty()) {
|
|
1460 |
fprintf(stderr, "Warning: Empty RSS_RULES_BASE key encountered\n");
|
|
1461 |
continue;
|
|
1462 |
}
|
|
1463 |
QStringList newValues;
|
|
1464 |
QStringList values = it.value();
|
|
1465 |
foreach(QString item, values) {
|
|
1466 |
// If there is no stringlist defined for a rule, use rule name directly
|
|
1467 |
// This is convenience for defining single line statements
|
|
1468 |
if (project->values(item).isEmpty()) {
|
|
1469 |
newValues << item;
|
|
1470 |
} else {
|
|
1471 |
foreach(QString itemRow, project->values(item)) {
|
|
1472 |
newValues << itemRow;
|
|
1473 |
}
|
|
1474 |
}
|
|
1475 |
}
|
|
1476 |
// Verify thet there is exactly one value in RSS_TAG_NBROFICONS
|
|
1477 |
if (newKey == RSS_TAG_NBROFICONS) {
|
|
1478 |
if (newValues.count() == 1) {
|
|
1479 |
numberOfIcons = newValues[0];
|
|
1480 |
} else {
|
|
1481 |
fprintf(stderr, "Warning: There must be exactly one value in '%s%s'\n",
|
|
1482 |
RSS_RULES_BASE, RSS_TAG_NBROFICONS);
|
|
1483 |
continue;
|
|
1484 |
}
|
|
1485 |
// Verify thet there is exactly one value in RSS_TAG_ICONFILE
|
|
1486 |
} else if (newKey == RSS_TAG_ICONFILE) {
|
|
1487 |
if (newValues.count() == 1) {
|
|
1488 |
iconFile = newValues[0];
|
|
1489 |
} else {
|
|
1490 |
fprintf(stderr, "Warning: There must be exactly one value in '%s%s'\n",
|
|
1491 |
RSS_RULES_BASE, RSS_TAG_ICONFILE);
|
|
1492 |
continue;
|
|
1493 |
}
|
|
1494 |
} else {
|
|
1495 |
fprintf(stderr, "Warning: Unsupported key:'%s%s'\n",
|
|
1496 |
RSS_RULES_BASE, newKey.toLatin1().constData());
|
|
1497 |
continue;
|
|
1498 |
}
|
|
1499 |
}
|
|
1500 |
}
|
|
1501 |
|
|
1502 |
foreach(QString item, project->values(RSS_RULES)) {
|
|
1503 |
// If there is no stringlist defined for a rule, use rule name directly
|
|
1504 |
// This is convenience for defining single line mmp statements
|
|
1505 |
if (project->values(item).isEmpty()) {
|
|
1506 |
userRssRules << item;
|
|
1507 |
} else {
|
|
1508 |
userRssRules << project->values(item);
|
|
1509 |
}
|
|
1510 |
}
|
|
1511 |
|
|
1512 |
// Validate that either both RSS_TAG_NBROFICONS and RSS_TAG_ICONFILE keys exist
|
|
1513 |
// or neither of them exist
|
|
1514 |
if (!((numberOfIcons.isEmpty() && iconFile.isEmpty()) ||
|
|
1515 |
(!numberOfIcons.isEmpty() && !iconFile.isEmpty()))) {
|
|
1516 |
numberOfIcons.clear();
|
|
1517 |
iconFile.clear();
|
|
1518 |
fprintf(stderr, "Warning: Both or neither of '%s%s' and '%s%s' keys must exist.\n",
|
|
1519 |
RSS_RULES_BASE, RSS_TAG_NBROFICONS, RSS_RULES_BASE, RSS_TAG_ICONFILE);
|
|
1520 |
}
|
|
1521 |
|
|
1522 |
// Validate that RSS_TAG_NBROFICONS contains only numbers
|
|
1523 |
if (!numberOfIcons.isEmpty()) {
|
|
1524 |
bool ok;
|
|
1525 |
numberOfIcons = numberOfIcons.simplified();
|
|
1526 |
int tmp = numberOfIcons.toInt(&ok);
|
|
1527 |
if (!ok) {
|
|
1528 |
numberOfIcons.clear();
|
|
1529 |
iconFile.clear();
|
|
1530 |
fprintf(stderr, "Warning: '%s%s' must be integer in decimal format.\n",
|
|
1531 |
RSS_RULES_BASE, RSS_TAG_NBROFICONS);
|
|
1532 |
}
|
|
1533 |
}
|
|
1534 |
}
|
|
1535 |
|
|
1536 |
QStringList SymbianMakefileGenerator::symbianLangCodesFromTsFiles()
|
|
1537 |
{
|
|
1538 |
QStringList tsfiles;
|
|
1539 |
QStringList symbianLangCodes;
|
|
1540 |
tsfiles << project->values("TRANSLATIONS");
|
|
1541 |
|
|
1542 |
fillQt2S60LangMapTable();
|
|
1543 |
|
|
1544 |
foreach(QString file, tsfiles) {
|
|
1545 |
int extIndex = file.lastIndexOf(".");
|
|
1546 |
int langIndex = file.lastIndexOf("_", (extIndex - file.length()));
|
|
1547 |
langIndex += 1;
|
|
1548 |
QString qtlang = file.mid(langIndex, extIndex - langIndex);
|
|
1549 |
QString s60lang = qt2S60LangMapTable.value(qtlang, QString("SC"));
|
|
1550 |
|
|
1551 |
if (!symbianLangCodes.contains(s60lang) && s60lang != "SC")
|
|
1552 |
symbianLangCodes += s60lang;
|
|
1553 |
}
|
|
1554 |
|
|
1555 |
return symbianLangCodes;
|
|
1556 |
}
|
|
1557 |
|
|
1558 |
void SymbianMakefileGenerator::fillQt2S60LangMapTable()
|
|
1559 |
{
|
|
1560 |
qt2S60LangMapTable.reserve(170); // 165 items at time of writing.
|
|
1561 |
qt2S60LangMapTable.insert("ab", "SC"); //Abkhazian //
|
|
1562 |
qt2S60LangMapTable.insert("om", "SC"); //Afan //
|
|
1563 |
qt2S60LangMapTable.insert("aa", "SC"); //Afar //
|
|
1564 |
qt2S60LangMapTable.insert("af", "34"); //Afrikaans //Afrikaans
|
|
1565 |
qt2S60LangMapTable.insert("sq", "35"); //Albanian //Albanian
|
|
1566 |
qt2S60LangMapTable.insert("am", "36"); //Amharic //Amharic
|
|
1567 |
qt2S60LangMapTable.insert("ar", "37"); //Arabic //Arabic
|
|
1568 |
qt2S60LangMapTable.insert("hy", "38"); //Armenian //Armenian
|
|
1569 |
qt2S60LangMapTable.insert("as", "SC"); //Assamese //
|
|
1570 |
qt2S60LangMapTable.insert("ay", "SC"); //Aymara //
|
|
1571 |
qt2S60LangMapTable.insert("az", "SC"); //Azerbaijani //
|
|
1572 |
qt2S60LangMapTable.insert("ba", "SC"); //Bashkir //
|
|
1573 |
qt2S60LangMapTable.insert("eu", "SC"); //Basque //
|
|
1574 |
qt2S60LangMapTable.insert("bn", "41"); //Bengali //Bengali
|
|
1575 |
qt2S60LangMapTable.insert("dz", "SC"); //Bhutani //
|
|
1576 |
qt2S60LangMapTable.insert("bh", "SC"); //Bihari //
|
|
1577 |
qt2S60LangMapTable.insert("bi", "SC"); //Bislama //
|
|
1578 |
qt2S60LangMapTable.insert("br", "SC"); //Breton //
|
|
1579 |
qt2S60LangMapTable.insert("bg", "42"); //Bulgarian //Bulgarian
|
|
1580 |
qt2S60LangMapTable.insert("my", "43"); //Burmese //Burmese
|
|
1581 |
qt2S60LangMapTable.insert("be", "40"); //Byelorussian //Belarussian
|
|
1582 |
qt2S60LangMapTable.insert("km", "SC"); //Cambodian //
|
|
1583 |
qt2S60LangMapTable.insert("ca", "44"); //Catalan //Catalan
|
|
1584 |
qt2S60LangMapTable.insert("zh", "SC"); //Chinese //
|
|
1585 |
qt2S60LangMapTable.insert("co", "SC"); //Corsican //
|
|
1586 |
qt2S60LangMapTable.insert("hr", "45"); //Croatian //Croatian
|
|
1587 |
qt2S60LangMapTable.insert("cs", "25"); //Czech //Czech
|
|
1588 |
qt2S60LangMapTable.insert("da", "07"); //Danish //Danish
|
|
1589 |
qt2S60LangMapTable.insert("nl", "18"); //Dutch //Dutch
|
|
1590 |
qt2S60LangMapTable.insert("en", "01"); //English //English(UK)
|
|
1591 |
qt2S60LangMapTable.insert("eo", "SC"); //Esperanto //
|
|
1592 |
qt2S60LangMapTable.insert("et", "49"); //Estonian //Estonian
|
|
1593 |
qt2S60LangMapTable.insert("fo", "SC"); //Faroese //
|
|
1594 |
qt2S60LangMapTable.insert("fj", "SC"); //Fiji //
|
|
1595 |
qt2S60LangMapTable.insert("fi", "09"); //Finnish //Finnish
|
|
1596 |
qt2S60LangMapTable.insert("fr", "02"); //French //French
|
|
1597 |
qt2S60LangMapTable.insert("fy", "SC"); //Frisian //
|
|
1598 |
qt2S60LangMapTable.insert("gd", "52"); //Gaelic //Gaelic
|
|
1599 |
qt2S60LangMapTable.insert("gl", "SC"); //Galician //
|
|
1600 |
qt2S60LangMapTable.insert("ka", "53"); //Georgian //Georgian
|
|
1601 |
qt2S60LangMapTable.insert("de", "03"); //German //German
|
|
1602 |
qt2S60LangMapTable.insert("el", "54"); //Greek //Greek
|
|
1603 |
qt2S60LangMapTable.insert("kl", "SC"); //Greenlandic //
|
|
1604 |
qt2S60LangMapTable.insert("gn", "SC"); //Guarani //
|
|
1605 |
qt2S60LangMapTable.insert("gu", "56"); //Gujarati //Gujarati
|
|
1606 |
qt2S60LangMapTable.insert("ha", "SC"); //Hausa //
|
|
1607 |
qt2S60LangMapTable.insert("he", "57"); //Hebrew //Hebrew
|
|
1608 |
qt2S60LangMapTable.insert("hi", "58"); //Hindi //Hindi
|
|
1609 |
qt2S60LangMapTable.insert("hu", "17"); //Hungarian //Hungarian
|
|
1610 |
qt2S60LangMapTable.insert("is", "15"); //Icelandic //Icelandic
|
|
1611 |
qt2S60LangMapTable.insert("id", "59"); //Indonesian //Indonesian
|
|
1612 |
qt2S60LangMapTable.insert("ia", "SC"); //Interlingua //
|
|
1613 |
qt2S60LangMapTable.insert("ie", "SC"); //Interlingue //
|
|
1614 |
qt2S60LangMapTable.insert("iu", "SC"); //Inuktitut //
|
|
1615 |
qt2S60LangMapTable.insert("ik", "SC"); //Inupiak //
|
|
1616 |
qt2S60LangMapTable.insert("ga", "60"); //Irish //Irish
|
|
1617 |
qt2S60LangMapTable.insert("it", "05"); //Italian //Italian
|
|
1618 |
qt2S60LangMapTable.insert("ja", "32"); //Japanese //Japanese
|
|
1619 |
qt2S60LangMapTable.insert("jv", "SC"); //Javanese //
|
|
1620 |
qt2S60LangMapTable.insert("kn", "62"); //Kannada //Kannada
|
|
1621 |
qt2S60LangMapTable.insert("ks", "SC"); //Kashmiri //
|
|
1622 |
qt2S60LangMapTable.insert("kk", "63"); //Kazakh //Kazakh
|
|
1623 |
qt2S60LangMapTable.insert("rw", "SC"); //Kinyarwanda //
|
|
1624 |
qt2S60LangMapTable.insert("ky", "SC"); //Kirghiz //
|
|
1625 |
qt2S60LangMapTable.insert("ko", "65"); //Korean //Korean
|
|
1626 |
qt2S60LangMapTable.insert("ku", "SC"); //Kurdish //
|
|
1627 |
qt2S60LangMapTable.insert("rn", "SC"); //Kurundi //
|
|
1628 |
qt2S60LangMapTable.insert("lo", "66"); //Laothian //Laothian
|
|
1629 |
qt2S60LangMapTable.insert("la", "SC"); //Latin //
|
|
1630 |
qt2S60LangMapTable.insert("lv", "67"); //Latvian //Latvian
|
|
1631 |
qt2S60LangMapTable.insert("ln", "SC"); //Lingala //
|
|
1632 |
qt2S60LangMapTable.insert("lt", "68"); //Lithuanian //Lithuanian
|
|
1633 |
qt2S60LangMapTable.insert("mk", "69"); //Macedonian //Macedonian
|
|
1634 |
qt2S60LangMapTable.insert("mg", "SC"); //Malagasy //
|
|
1635 |
qt2S60LangMapTable.insert("ms", "70"); //Malay //Malay
|
|
1636 |
qt2S60LangMapTable.insert("ml", "71"); //Malayalam //Malayalam
|
|
1637 |
qt2S60LangMapTable.insert("mt", "SC"); //Maltese //
|
|
1638 |
qt2S60LangMapTable.insert("mi", "SC"); //Maori //
|
|
1639 |
qt2S60LangMapTable.insert("mr", "72"); //Marathi //Marathi
|
|
1640 |
qt2S60LangMapTable.insert("mo", "73"); //Moldavian //Moldovian
|
|
1641 |
qt2S60LangMapTable.insert("mn", "74"); //Mongolian //Mongolian
|
|
1642 |
qt2S60LangMapTable.insert("na", "SC"); //Nauru //
|
|
1643 |
qt2S60LangMapTable.insert("ne", "SC"); //Nepali //
|
|
1644 |
qt2S60LangMapTable.insert("nb", "08"); //Norwegian //Norwegian
|
|
1645 |
qt2S60LangMapTable.insert("oc", "SC"); //Occitan //
|
|
1646 |
qt2S60LangMapTable.insert("or", "SC"); //Oriya //
|
|
1647 |
qt2S60LangMapTable.insert("ps", "SC"); //Pashto //
|
|
1648 |
qt2S60LangMapTable.insert("fa", "SC"); //Persian //
|
|
1649 |
qt2S60LangMapTable.insert("pl", "27"); //Polish //Polish
|
|
1650 |
qt2S60LangMapTable.insert("pt", "13"); //Portuguese //Portuguese
|
|
1651 |
qt2S60LangMapTable.insert("pa", "77"); //Punjabi //Punjabi
|
|
1652 |
qt2S60LangMapTable.insert("qu", "SC"); //Quechua //
|
|
1653 |
qt2S60LangMapTable.insert("rm", "SC"); //RhaetoRomance //
|
|
1654 |
qt2S60LangMapTable.insert("ro", "78"); //Romanian //Romanian
|
|
1655 |
qt2S60LangMapTable.insert("ru", "16"); //Russian //Russian
|
|
1656 |
qt2S60LangMapTable.insert("sm", "SC"); //Samoan //
|
|
1657 |
qt2S60LangMapTable.insert("sg", "SC"); //Sangho //
|
|
1658 |
qt2S60LangMapTable.insert("sa", "SC"); //Sanskrit //
|
|
1659 |
qt2S60LangMapTable.insert("sr", "79"); //Serbian //Serbian
|
|
1660 |
qt2S60LangMapTable.insert("sh", "SC"); //SerboCroatian //
|
|
1661 |
qt2S60LangMapTable.insert("st", "SC"); //Sesotho //
|
|
1662 |
qt2S60LangMapTable.insert("tn", "SC"); //Setswana //
|
|
1663 |
qt2S60LangMapTable.insert("sn", "SC"); //Shona //
|
|
1664 |
qt2S60LangMapTable.insert("sd", "SC"); //Sindhi //
|
|
1665 |
qt2S60LangMapTable.insert("si", "80"); //Singhalese //Sinhalese
|
|
1666 |
qt2S60LangMapTable.insert("ss", "SC"); //Siswati //
|
|
1667 |
qt2S60LangMapTable.insert("sk", "26"); //Slovak //Slovak
|
|
1668 |
qt2S60LangMapTable.insert("sl", "28"); //Slovenian //Slovenian
|
|
1669 |
qt2S60LangMapTable.insert("so", "81"); //Somali //Somali
|
|
1670 |
qt2S60LangMapTable.insert("es", "04"); //Spanish //Spanish
|
|
1671 |
qt2S60LangMapTable.insert("su", "SC"); //Sundanese //
|
|
1672 |
qt2S60LangMapTable.insert("sw", "84"); //Swahili //Swahili
|
|
1673 |
qt2S60LangMapTable.insert("sv", "06"); //Swedish //Swedish
|
|
1674 |
qt2S60LangMapTable.insert("tl", "39"); //Tagalog //Tagalog
|
|
1675 |
qt2S60LangMapTable.insert("tg", "SC"); //Tajik //
|
|
1676 |
qt2S60LangMapTable.insert("ta", "87"); //Tamil //Tamil
|
|
1677 |
qt2S60LangMapTable.insert("tt", "SC"); //Tatar //
|
|
1678 |
qt2S60LangMapTable.insert("te", "88"); //Telugu //Telugu
|
|
1679 |
qt2S60LangMapTable.insert("th", "33"); //Thai //Thai
|
|
1680 |
qt2S60LangMapTable.insert("bo", "89"); //Tibetan //Tibetan
|
|
1681 |
qt2S60LangMapTable.insert("ti", "90"); //Tigrinya //Tigrinya
|
|
1682 |
qt2S60LangMapTable.insert("to", "SC"); //Tonga //
|
|
1683 |
qt2S60LangMapTable.insert("ts", "SC"); //Tsonga //
|
|
1684 |
qt2S60LangMapTable.insert("tr", "14"); //Turkish //Turkish
|
|
1685 |
qt2S60LangMapTable.insert("tk", "92"); //Turkmen //Turkmen
|
|
1686 |
qt2S60LangMapTable.insert("tw", "SC"); //Twi //
|
|
1687 |
qt2S60LangMapTable.insert("ug", "SC"); //Uigur //
|
|
1688 |
qt2S60LangMapTable.insert("uk", "93"); //Ukrainian //Ukrainian
|
|
1689 |
qt2S60LangMapTable.insert("ur", "94"); //Urdu //Urdu
|
|
1690 |
qt2S60LangMapTable.insert("uz", "SC"); //Uzbek //
|
|
1691 |
qt2S60LangMapTable.insert("vi", "96"); //Vietnamese //Vietnamese
|
|
1692 |
qt2S60LangMapTable.insert("vo", "SC"); //Volapuk //
|
|
1693 |
qt2S60LangMapTable.insert("cy", "97"); //Welsh //Welsh
|
|
1694 |
qt2S60LangMapTable.insert("wo", "SC"); //Wolof //
|
|
1695 |
qt2S60LangMapTable.insert("xh", "SC"); //Xhosa //
|
|
1696 |
qt2S60LangMapTable.insert("yi", "SC"); //Yiddish //
|
|
1697 |
qt2S60LangMapTable.insert("yo", "SC"); //Yoruba //
|
|
1698 |
qt2S60LangMapTable.insert("za", "SC"); //Zhuang //
|
|
1699 |
qt2S60LangMapTable.insert("zu", "98"); //Zulu //Zulu
|
|
1700 |
qt2S60LangMapTable.insert("nn", "75"); //Nynorsk //NorwegianNynorsk
|
|
1701 |
qt2S60LangMapTable.insert("bs", "SC"); //Bosnian //
|
|
1702 |
qt2S60LangMapTable.insert("dv", "SC"); //Divehi //
|
|
1703 |
qt2S60LangMapTable.insert("gv", "SC"); //Manx //
|
|
1704 |
qt2S60LangMapTable.insert("kw", "SC"); //Cornish //
|
|
1705 |
qt2S60LangMapTable.insert("ak", "SC"); //Akan //
|
|
1706 |
qt2S60LangMapTable.insert("kok", "SC"); //Konkani //
|
|
1707 |
qt2S60LangMapTable.insert("gaa", "SC"); //Ga //
|
|
1708 |
qt2S60LangMapTable.insert("ig", "SC"); //Igbo //
|
|
1709 |
qt2S60LangMapTable.insert("kam", "SC"); //Kamba //
|
|
1710 |
qt2S60LangMapTable.insert("syr", "SC"); //Syriac //
|
|
1711 |
qt2S60LangMapTable.insert("byn", "SC"); //Blin //
|
|
1712 |
qt2S60LangMapTable.insert("gez", "SC"); //Geez //
|
|
1713 |
qt2S60LangMapTable.insert("kfo", "SC"); //Koro //
|
|
1714 |
qt2S60LangMapTable.insert("sid", "SC"); //Sidamo //
|
|
1715 |
qt2S60LangMapTable.insert("cch", "SC"); //Atsam //
|
|
1716 |
qt2S60LangMapTable.insert("tig", "SC"); //Tigre //
|
|
1717 |
qt2S60LangMapTable.insert("kaj", "SC"); //Jju //
|
|
1718 |
qt2S60LangMapTable.insert("fur", "SC"); //Friulian //
|
|
1719 |
qt2S60LangMapTable.insert("ve", "SC"); //Venda //
|
|
1720 |
qt2S60LangMapTable.insert("ee", "SC"); //Ewe //
|
|
1721 |
qt2S60LangMapTable.insert("wa", "SC"); //Walamo //
|
|
1722 |
qt2S60LangMapTable.insert("haw", "SC"); //Hawaiian //
|
|
1723 |
qt2S60LangMapTable.insert("kcg", "SC"); //Tyap //
|
|
1724 |
qt2S60LangMapTable.insert("ny", "SC"); //Chewa //
|
|
1725 |
}
|
|
1726 |
|
|
1727 |
void SymbianMakefileGenerator::appendIfnotExist(QStringList &list, QString value)
|
|
1728 |
{
|
|
1729 |
if (!list.contains(value))
|
|
1730 |
list += value;
|
|
1731 |
}
|
|
1732 |
|
|
1733 |
void SymbianMakefileGenerator::appendIfnotExist(QStringList &list, QStringList values)
|
|
1734 |
{
|
|
1735 |
foreach(QString item, values)
|
|
1736 |
appendIfnotExist(list, item);
|
|
1737 |
}
|
|
1738 |
|
|
1739 |
QString SymbianMakefileGenerator::removePathSeparators(QString &file)
|
|
1740 |
{
|
|
1741 |
QString ret = file;
|
|
1742 |
while (ret.indexOf(QDir::separator()) > 0) {
|
|
1743 |
ret.remove(0, ret.indexOf(QDir::separator()) + 1);
|
|
1744 |
}
|
|
1745 |
|
|
1746 |
return ret;
|
|
1747 |
}
|
|
1748 |
|
|
1749 |
|
|
1750 |
QString SymbianMakefileGenerator::removeTrailingPathSeparators(QString &file)
|
|
1751 |
{
|
|
1752 |
QString ret = file;
|
|
1753 |
if (ret.endsWith(QDir::separator())) {
|
|
1754 |
ret.remove(ret.length() - 1, 1);
|
|
1755 |
}
|
|
1756 |
|
|
1757 |
return ret;
|
|
1758 |
}
|
|
1759 |
|
|
1760 |
void SymbianMakefileGenerator::generateCleanCommands(QTextStream& t,
|
|
1761 |
const QStringList& toClean,
|
|
1762 |
const QString& cmd,
|
|
1763 |
const QString& cmdOptions,
|
|
1764 |
const QString& itemPrefix,
|
|
1765 |
const QString& itemSuffix)
|
|
1766 |
{
|
|
1767 |
for (int i = 0; i < toClean.size(); ++i) {
|
|
1768 |
QString item = toClean.at(i);
|
|
1769 |
item.prepend(itemPrefix).append(itemSuffix);
|
|
1770 |
#if defined(Q_OS_WIN)
|
|
1771 |
t << "\t-@ if EXIST \"" << QDir::toNativeSeparators(item) << "\" ";
|
|
1772 |
t << cmd << " " << cmdOptions << " \"" << QDir::toNativeSeparators(item) << "\"" << endl;
|
|
1773 |
#else
|
|
1774 |
t << "\t-if test -f " << QDir::toNativeSeparators(item) << "; then ";
|
|
1775 |
t << cmd << " " << cmdOptions << " " << QDir::toNativeSeparators(item) << "; fi" << endl;
|
|
1776 |
#endif
|
|
1777 |
}
|
|
1778 |
}
|
|
1779 |
|
|
1780 |
void SymbianMakefileGenerator::removeSpecialCharacters(QString& str)
|
|
1781 |
{
|
|
1782 |
str.replace(QString("/"), QString("_"));
|
|
1783 |
str.replace(QString("\\"), QString("_"));
|
|
1784 |
str.replace(QString("-"), QString("_"));
|
|
1785 |
str.replace(QString(":"), QString("_"));
|
|
1786 |
str.replace(QString("."), QString("_"));
|
|
1787 |
str.replace(QString(" "), QString("_"));
|
|
1788 |
}
|
|
1789 |
|
|
1790 |
void SymbianMakefileGenerator::writeSisTargets(QTextStream &t)
|
|
1791 |
{
|
|
1792 |
t << SIS_TARGET ": " RESTORE_BUILD_TARGET << endl;
|
|
1793 |
QString siscommand = QString("\t$(if $(wildcard %1_template.%2),$(if $(wildcard %3)," \
|
|
1794 |
"$(MAKE) -s -f $(MAKEFILE) %4,$(MAKE) -s -f $(MAKEFILE) %5)," \
|
|
1795 |
"$(MAKE) -s -f $(MAKEFILE) %6)")
|
|
1796 |
.arg(fixedTarget)
|
|
1797 |
.arg("pkg")
|
|
1798 |
.arg(MAKE_CACHE_NAME)
|
|
1799 |
.arg(OK_SIS_TARGET)
|
|
1800 |
.arg(FAIL_SIS_NOCACHE_TARGET)
|
|
1801 |
.arg(FAIL_SIS_NOPKG_TARGET);
|
|
1802 |
t << siscommand << endl;
|
|
1803 |
t << endl;
|
|
1804 |
|
|
1805 |
t << OK_SIS_TARGET ":" << endl;
|
|
1806 |
|
|
1807 |
QString pkgcommand = QString("\tcreatepackage.bat $(QT_SIS_OPTIONS) %1_template.%2 $(QT_SIS_TARGET) " \
|
|
1808 |
"$(QT_SIS_CERTIFICATE) $(QT_SIS_KEY) $(QT_SIS_PASSPHRASE)")
|
|
1809 |
.arg(fixedTarget)
|
|
1810 |
.arg("pkg");
|
|
1811 |
t << pkgcommand << endl;
|
|
1812 |
t << endl;
|
|
1813 |
|
|
1814 |
t << FAIL_SIS_NOPKG_TARGET ":" << endl;
|
|
1815 |
t << "\t$(error PKG file does not exist, 'SIS' target is only supported for executables or projects with DEPLOYMENT statement)" << endl;
|
|
1816 |
t << endl;
|
|
1817 |
|
|
1818 |
t << FAIL_SIS_NOCACHE_TARGET ":" << endl;
|
|
1819 |
t << "\t$(error Project has to be build before calling 'SIS' target)" << endl;
|
|
1820 |
t << endl;
|
|
1821 |
|
|
1822 |
|
|
1823 |
t << RESTORE_BUILD_TARGET ":" << endl;
|
|
1824 |
t << "-include " MAKE_CACHE_NAME << endl;
|
|
1825 |
t << endl;
|
|
1826 |
}
|
|
1827 |
|
|
1828 |
void SymbianMakefileGenerator::generateDistcleanTargets(QTextStream& t)
|
|
1829 |
{
|
|
1830 |
t << "dodistclean:" << endl;
|
|
1831 |
const QStringList &subdirs = project->values("SUBDIRS");
|
|
1832 |
foreach(QString item, subdirs) {
|
|
1833 |
bool fromFile = false;
|
|
1834 |
QString fixedItem;
|
|
1835 |
if (!project->isEmpty(item + ".file")) {
|
|
1836 |
fixedItem = project->first(item + ".file");
|
|
1837 |
fromFile = true;
|
|
1838 |
} else if (!project->isEmpty(item + ".subdir")) {
|
|
1839 |
fixedItem = project->first(item + ".subdir");
|
|
1840 |
fromFile = false;
|
|
1841 |
} else {
|
|
1842 |
fromFile = item.endsWith(Option::pro_ext);
|
|
1843 |
fixedItem = item;
|
|
1844 |
}
|
|
1845 |
QFileInfo fi(fileInfo(fixedItem));
|
|
1846 |
if (!fromFile) {
|
|
1847 |
t << "\t-$(MAKE) -f \"" << Option::fixPathToTargetOS(fi.absoluteFilePath() + "/Makefile") << "\" dodistclean" << endl;
|
|
1848 |
} else {
|
|
1849 |
QString itemName = fi.fileName();
|
|
1850 |
int extIndex = itemName.lastIndexOf(Option::pro_ext);
|
|
1851 |
if (extIndex)
|
|
1852 |
fixedItem = fi.absolutePath() + "/" + QString("Makefile.") + itemName.mid(0, extIndex);
|
|
1853 |
t << "\t-$(MAKE) -f \"" << Option::fixPathToTargetOS(fixedItem) << "\" dodistclean" << endl;
|
|
1854 |
}
|
|
1855 |
|
|
1856 |
}
|
|
1857 |
|
|
1858 |
generatedFiles << Option::fixPathToTargetOS(fileInfo(Option::output.fileName()).absoluteFilePath()); // bld.inf
|
|
1859 |
generatedFiles << project->values("QMAKE_INTERNAL_PRL_FILE"); // Add generated prl files for cleanup
|
|
1860 |
generatedFiles << project->values("QMAKE_DISTCLEAN"); // Add any additional files marked for distclean
|
|
1861 |
QStringList fixedFiles;
|
|
1862 |
QStringList fixedDirs;
|
|
1863 |
foreach(QString item, generatedFiles) {
|
|
1864 |
QString fixedItem = Option::fixPathToTargetOS(fileInfo(item).absoluteFilePath());
|
|
1865 |
if (!fixedFiles.contains(fixedItem)) {
|
|
1866 |
fixedFiles << fixedItem;
|
|
1867 |
}
|
|
1868 |
}
|
|
1869 |
foreach(QString item, generatedDirs) {
|
|
1870 |
QString fixedItem = Option::fixPathToTargetOS(fileInfo(item).absoluteFilePath());
|
|
1871 |
if (!fixedDirs.contains(fixedItem)) {
|
|
1872 |
fixedDirs << fixedItem;
|
|
1873 |
}
|
|
1874 |
}
|
|
1875 |
generateCleanCommands(t, fixedFiles, "$(DEL_FILE)", "", "", "");
|
|
1876 |
generateCleanCommands(t, fixedDirs, "$(DEL_DIR)", "", "", "");
|
|
1877 |
t << endl;
|
|
1878 |
|
|
1879 |
t << "distclean: clean dodistclean" << endl;
|
|
1880 |
t << endl;
|
|
1881 |
}
|