|
1 # |
|
2 # Copyright (c) 2007-2009 Nokia Corporation and/or its subsidiary(-ies). |
|
3 # All rights reserved. |
|
4 # This component and the accompanying materials are made available |
|
5 # under the terms of the License "Eclipse Public License v1.0" |
|
6 # which accompanies this distribution, and is available |
|
7 # at the URL "http://www.eclipse.org/legal/epl-v10.html". |
|
8 # |
|
9 # Initial Contributors: |
|
10 # Nokia Corporation - initial contribution. |
|
11 # |
|
12 # Contributors: |
|
13 # |
|
14 # Description: |
|
15 # raptor_utilities module |
|
16 # Useful wrapper functions and classes used in Raptor processing |
|
17 # |
|
18 |
|
19 import generic_path |
|
20 import os.path |
|
21 import re |
|
22 import sys |
|
23 |
|
24 dosSlashRegEx = re.compile(r'\\') |
|
25 unixSlashRegEx = re.compile(r'/') |
|
26 dosDriveRegEx = re.compile("^([A-Za-z]{1}):") |
|
27 |
|
28 def getOSPlatform(): |
|
29 return sys.platform.lower() |
|
30 |
|
31 def getOSFileSystem(): |
|
32 if getOSPlatform().startswith("win"): |
|
33 return "cygwin" |
|
34 else: |
|
35 return "unix" |
|
36 |
|
37 def convertToUnixSlash(aReference): |
|
38 return dosSlashRegEx.sub(r'/', aReference) |
|
39 |
|
40 def convertToDOSSlash(aReference): |
|
41 return unixSlashRegEx.sub(r'\\', aReference) |
|
42 |
|
43 def absPathFromPath(aPathRoot, aReference): |
|
44 pathRoot = convertToUnixSlash(aPathRoot) |
|
45 reference = convertToUnixSlash(aReference) |
|
46 |
|
47 if os.path.isabs(reference): |
|
48 reference = reference.lstrip(r'/') |
|
49 |
|
50 joined = os.path.join(pathRoot, reference) |
|
51 |
|
52 return os.path.abspath(joined) |
|
53 |
|
54 |
|
55 def absPathFromFile(aFileRoot, aReference): |
|
56 pathRoot = os.path.dirname(aFileRoot) |
|
57 return absPathFromPath(pathRoot, aReference) |
|
58 |
|
59 def sanitise(aPotentialFilename): |
|
60 "Take a string and return a version suitable for use as a filename." |
|
61 return re.sub("(\\\\|\/|:|;| )", "_", aPotentialFilename) |
|
62 |
|
63 def resolveSymbianPath(aFileRoot, aReference, aMainType="", aSubType="", aEPOCROOT="$(EPOCROOT)"): |
|
64 """ Convert raw Symbian metadata path/file references into absolute makefile references, or list of references |
|
65 |
|
66 <drive>-prefix : maps to an emulated drive depending on the following cases: |
|
67 (a) If the drive is C:, it maps to the *two* locations |
|
68 $(EPOCROOT)/epoc32/data/<drive>/<path> and |
|
69 $(EPOCROOT)/epoc32/winscw/<drive>/<path> |
|
70 (b) If the drive is A:, B:, or D: to Z:, it maps to the *three* locations |
|
71 $(EPOCROOT)/epoc32/data/<drive>/<path> and |
|
72 $(EPOCROOT)/epoc32/release/winscw/udeb/<drive>/<path> and |
|
73 $(EPOCROOT)/epoc32/release/winscw/urel/<drive>/<path> |
|
74 Absolute : true absolute if: |
|
75 (a) PRJ_*EXPORTS destination or DEFFILE location and |
|
76 (b) not starting with an 'epoc32' |
|
77 otherwise relative to $(EPOCROOT) |
|
78 Relative : relative to $(EPOCROOT)/epoc32/include if: |
|
79 (a) PRJ_EXPORTS destination and |
|
80 (b) not a :zip statement, |
|
81 relative to $(EPOCROOT) if: |
|
82 (a) PRJ_(TEST)EXPORTS destination and |
|
83 (b) a :zip statement, |
|
84 otherwise relative to aFileRoot |
|
85 |-prefix : relative to aFileRoot |
|
86 +-prefix : relative to $(EPOCROOT)/epoc32""" |
|
87 |
|
88 # Both reference and fileroot can have backslashes - so convert them. |
|
89 reference = convertToUnixSlash(aReference) |
|
90 fileroot = convertToUnixSlash(aFileRoot) |
|
91 |
|
92 # Remove Trailing backslashes so that the expansions doesnt mess up the shell |
|
93 if reference.endswith('/') and len(reference) > 1: |
|
94 reference = reference.rstrip('/') |
|
95 |
|
96 emulatedDrive = dosDriveRegEx.match(reference) |
|
97 if emulatedDrive: |
|
98 # Emulated drive C:/ Z:/ and the like |
|
99 # C: drive |
|
100 if reference.lower().startswith("c"): |
|
101 resolvedPath = [] |
|
102 resolvedPath.append(dosDriveRegEx.sub(aEPOCROOT+'/epoc32/data/'+emulatedDrive.group(1), reference)) |
|
103 resolvedPath.append(dosDriveRegEx.sub(aEPOCROOT+'/epoc32/winscw/'+emulatedDrive.group(1), reference)) |
|
104 else: # Other letters: A, B and D to Z |
|
105 resolvedPath = [] |
|
106 resolvedPath.append(dosDriveRegEx.sub(aEPOCROOT+'/epoc32/data/'+emulatedDrive.group(1), reference)) |
|
107 resolvedPath.append(dosDriveRegEx.sub(aEPOCROOT+'/epoc32/release/winscw/udeb/'+emulatedDrive.group(1), reference)) |
|
108 resolvedPath.append(dosDriveRegEx.sub(aEPOCROOT+'/epoc32/release/winscw/urel/'+emulatedDrive.group(1), reference)) |
|
109 elif os.path.isabs(reference): |
|
110 # Absolute |
|
111 if re.search("(DEFFILE|PRJ_(TEST)?EXPORTS)", aMainType, re.I) and not re.search("^\/epoc32\/", reference, re.I): |
|
112 # Ensures prepending of drive if on Windows |
|
113 resolvedPath = os.path.abspath(reference) |
|
114 else: |
|
115 resolvedPath = aEPOCROOT + reference |
|
116 |
|
117 elif reference.startswith("+"): |
|
118 # '+' prefix |
|
119 reference = reference.lstrip(r'+') |
|
120 resolvedPath = aEPOCROOT + '/epoc32'+reference |
|
121 elif reference.startswith("|"): |
|
122 # '|' prefix |
|
123 reference = reference.lstrip(r'|') |
|
124 resolvedPath = absPathFromFile(fileroot, reference) |
|
125 else: |
|
126 # Relative |
|
127 if aMainType == "PRJ_EXPORTS" and aSubType != ":zip": |
|
128 resolvedPath = aEPOCROOT + '/epoc32/include/'+reference |
|
129 elif aSubType == ":zip": |
|
130 resolvedPath = aEPOCROOT + '/' + reference |
|
131 else: |
|
132 resolvedPath = absPathFromFile(fileroot, aReference) |
|
133 |
|
134 if isinstance(resolvedPath, list): |
|
135 # In this case, this is a list of export destinations, |
|
136 makefilePath = map(lambda x: str(generic_path.Path(x)), resolvedPath) |
|
137 else: |
|
138 makefilePath = str(generic_path.Path(resolvedPath)) |
|
139 |
|
140 return makefilePath # Note this is either a list of strings, or a single string |
|
141 |
|
142 |
|
143 class ExternalTool(object): |
|
144 """ Generic wrapper for an external tool |
|
145 |
|
146 Provides the basic means to wrap up a tool that is external to Raptor with a |
|
147 consistent interface for both invocation and the capture of output.""" |
|
148 |
|
149 def __init__(self, aTool): |
|
150 self.__Tool = aTool |
|
151 self.__Output = [] |
|
152 |
|
153 def call(self, aArgs): |
|
154 print "RUNNNING: %s %s" %(self.__Tool, aArgs) |
|
155 (input, output) = os.popen2(self.__Tool + " " + aArgs) |
|
156 self.__Output = output.read() |
|
157 return output.close() |
|
158 |
|
159 def getTool(self): |
|
160 return self.__Tool |
|
161 |
|
162 def getOutput(self): |
|
163 return self.__Output |
|
164 |
|
165 def getOutputLines(self): |
|
166 return self.__Output.split("\n") |
|
167 |
|
168 |
|
169 class NullLog(object): |
|
170 """ If your class has these methods then it can act as a log """ |
|
171 def Info(self, format, *extras): |
|
172 "Send an information message to the configured channel" |
|
173 return |
|
174 |
|
175 def ClockInfo(self): |
|
176 "Print a timestamp in seconds" |
|
177 return |
|
178 |
|
179 def Debug(self, format, *extras): |
|
180 "Send a debugging message to the configured channel" |
|
181 return |
|
182 |
|
183 def Warn(self, format, *extras): |
|
184 "Send a warning message to the configured channel" |
|
185 return |
|
186 |
|
187 def Error(self, format, *extras): |
|
188 "Send an error message to the configured channel" |
|
189 return |
|
190 |
|
191 nulllog = NullLog() |