blob: 20582a2b2675598f5acf02670cee9744dd065f31 [file] [log] [blame]
Daniel Dunbar768be1c2009-01-21 02:03:52 +00001import os
2
Daniel Dunbar11672ec2009-01-13 18:51:26 +00003import Arguments
Daniel Dunbar43124722009-01-10 02:07:54 +00004import Phases
5import Tools
Daniel Dunbar33a5d612009-01-14 01:03:36 +00006import Types
Daniel Dunbar43124722009-01-10 02:07:54 +00007
8###
9
10class ToolChain(object):
11 """ToolChain - Provide mappings of Actions to Tools."""
12
Daniel Dunbar768be1c2009-01-21 02:03:52 +000013 def __init__(self, driver,
14 filePathPrefixes=[],
15 programPathPrefixes=[]):
Daniel Dunbar43124722009-01-10 02:07:54 +000016 self.driver = driver
Daniel Dunbar768be1c2009-01-21 02:03:52 +000017 self.filePathPrefixes = list(filePathPrefixes)
18 self.programPathPrefixes = list(programPathPrefixes)
19
20 def getFilePath(self, name):
21 return self.driver.getFilePath(name, self)
22
23 def getProgramPath(self, name):
24 return self.driver.getProgramPath(name, self)
Daniel Dunbar43124722009-01-10 02:07:54 +000025
26 def selectTool(self, action):
27 """selectTool - Return a Tool instance to use for handling
28 some particular action."""
29 abstract
30
Daniel Dunbar11672ec2009-01-13 18:51:26 +000031 def translateArgs(self, args, arch):
32 """translateArgs - Callback to allow argument translation for
33 an entire toolchain."""
34
35 # FIXME: Would be nice to move arch handling out of generic
36 # code.
37 if arch:
38 archName = args.getValue(arch)
39 al = Arguments.DerivedArgList(args)
40 for arg in args.args:
41 if arg.opt is args.parser.archOption:
42 if arg is arch:
43 al.append(arg)
44 elif arg.opt is args.parser.XarchOption:
45 if args.getJoinedValue(arg) == archName:
46 # FIXME: Fix this.
47 arg = args.parser.lookupOptForArg(Arguments.InputIndex(0, arg.index.pos + 1),
48 args.getSeparateValue(arg),
49 iter([]))
50 al.append(arg)
51 else:
52 al.append(arg)
53 return al
54 else:
55 return args
56
Daniel Dunbar9c257c32009-01-12 04:21:12 +000057class Darwin_X86_ToolChain(ToolChain):
Daniel Dunbar33a5d612009-01-14 01:03:36 +000058 def __init__(self, driver, darwinVersion, gccVersion, archName):
Daniel Dunbar9c257c32009-01-12 04:21:12 +000059 super(Darwin_X86_ToolChain, self).__init__(driver)
Daniel Dunbar9cb22532009-01-12 07:45:49 +000060 assert isinstance(darwinVersion, tuple) and len(darwinVersion) == 3
61 assert isinstance(gccVersion, tuple) and len(gccVersion) == 3
62 self.darwinVersion = darwinVersion
63 self.gccVersion = gccVersion
Daniel Dunbar33a5d612009-01-14 01:03:36 +000064 self.archName = archName
Daniel Dunbar9cb22532009-01-12 07:45:49 +000065
Daniel Dunbarde388a52009-01-21 01:07:49 +000066 self.clangTool = Tools.Clang_CompileTool(self)
Daniel Dunbar43124722009-01-10 02:07:54 +000067 self.toolMap = {
Daniel Dunbar06172d62009-01-20 00:47:24 +000068 Phases.PreprocessPhase : Tools.Darwin_X86_PreprocessTool(self),
Daniel Dunbarde388a52009-01-21 01:07:49 +000069 Phases.AnalyzePhase : self.clangTool,
70 Phases.SyntaxOnlyPhase : Tools.Darwin_X86_CompileTool(self),
Daniel Dunbar11672ec2009-01-13 18:51:26 +000071 Phases.CompilePhase : Tools.Darwin_X86_CompileTool(self),
Daniel Dunbar4e3d7622009-01-20 05:51:52 +000072 Phases.PrecompilePhase : Tools.Darwin_X86_CompileTool(self),
Daniel Dunbar9cb22532009-01-12 07:45:49 +000073 Phases.AssemblePhase : Tools.Darwin_AssembleTool(self),
74 Phases.LinkPhase : Tools.Darwin_X86_LinkTool(self),
Daniel Dunbar43124722009-01-10 02:07:54 +000075 Phases.LipoPhase : Tools.LipoTool(),
76 }
77
Daniel Dunbar768be1c2009-01-21 02:03:52 +000078 self.filePathPrefixes.append(os.path.join(self.driver.driverDir,
79 '../lib/gcc',
80 self.getToolChainDir()))
81 self.filePathPrefixes.append(os.path.join(self.driver.driverDir,
82 '/usr/lib/gcc',
83 self.getToolChainDir()))
84
85 self.programPathPrefixes.append(os.path.join(self.driver.driverDir,
86 '../libexec/gcc',
87 self.getToolChainDir()))
88 self.programPathPrefixes.append(os.path.join(self.driver.driverDir,
89 '/usr/libexec/gcc',
90 self.getToolChainDir()))
91 self.programPathPrefixes.append(self.driver.driverDir)
92
Daniel Dunbar9cb22532009-01-12 07:45:49 +000093 def getToolChainDir(self):
94 return 'i686-apple-darwin%d/%s' % (self.darwinVersion[0],
95 '.'.join(map(str,self.gccVersion)))
96
Daniel Dunbardff9f502009-01-12 18:51:02 +000097 def getMacosxVersionMin(self):
98 major,minor,minorminor = self.darwinVersion
99 return '%d.%d.%d' % (10, major-4, minor)
100
Daniel Dunbar43124722009-01-10 02:07:54 +0000101 def selectTool(self, action):
102 assert isinstance(action, Phases.JobAction)
Daniel Dunbar33a5d612009-01-14 01:03:36 +0000103
Daniel Dunbar89db21f2009-01-14 23:26:40 +0000104 if self.driver.cccClang and self.archName == 'i386':
105 if (action.inputs[0].type in (Types.CType, Types.CTypeNoPP,
106 Types.ObjCType, Types.ObjCTypeNoPP) and
107 isinstance(action.phase, Phases.CompilePhase)):
108 return self.clangTool
109 elif (action.inputs[0].type in (Types.CHeaderType, Types.CHeaderNoPPType,
110 Types.ObjCHeaderType, Types.ObjCHeaderNoPPType) and
111 isinstance(action.phase, Phases.PrecompilePhase)):
112 return self.clangTool
Daniel Dunbar33a5d612009-01-14 01:03:36 +0000113
Daniel Dunbar43124722009-01-10 02:07:54 +0000114 return self.toolMap[action.phase.__class__]
115
Daniel Dunbar25d4a8f2009-01-13 21:07:43 +0000116 def translateArgs(self, args, arch):
117 args = super(Darwin_X86_ToolChain, self).translateArgs(args, arch)
118
119 # If arch hasn't been bound we don't need to do anything yet.
120 if not arch:
121 return args
122
Daniel Dunbar9c3ea762009-01-16 20:25:36 +0000123 # FIXME: We really want to get out of the tool chain level
124 # argument translation business, as it makes the driver
125 # functionality much more opaque. For now, we follow gcc
126 # closely solely for the purpose of easily achieving feature
127 # parity & testability. Once we have something that works, we
128 # should reevaluate each translation and try to push it down
129 # into tool specific logic.
130
Daniel Dunbar25d4a8f2009-01-13 21:07:43 +0000131 al = Arguments.DerivedArgList(args)
132 if not args.getLastArg(args.parser.m_macosxVersionMinOption):
133 al.append(al.makeJoinedArg(self.getMacosxVersionMin(),
134 args.parser.m_macosxVersionMinOption))
135 for arg in args:
Daniel Dunbar9c3ea762009-01-16 20:25:36 +0000136 # Sob. These is strictly gcc compatible for the time
137 # being. Apple gcc translates options twice, which means
138 # that self-expanding options add duplicates.
139 if arg.opt is args.parser.m_kernelOption:
140 al.append(arg)
141 al.append(al.makeFlagArg(args.parser.staticOption))
142 al.append(al.makeFlagArg(args.parser.staticOption))
143 elif arg.opt is args.parser.dependencyFileOption:
144 al.append(al.makeSeparateArg(args.getValue(arg),
145 args.parser.MFOption))
146 elif arg.opt is args.parser.gfullOption:
147 al.append(al.makeFlagArg(args.parser.gOption))
148 al.append(al.makeFlagArg(args.parser.f_noEliminateUnusedDebugSymbolsOption))
149 elif arg.opt is args.parser.gusedOption:
150 al.append(al.makeFlagArg(args.parser.gOption))
151 al.append(al.makeFlagArg(args.parser.f_eliminateUnusedDebugSymbolsOption))
152 elif arg.opt is args.parser.f_appleKextOption:
153 al.append(arg)
154 al.append(al.makeFlagArg(args.parser.staticOption))
155 al.append(al.makeFlagArg(args.parser.staticOption))
156 elif arg.opt is args.parser.f_terminatedVtablesOption:
157 al.append(al.makeFlagArg(args.parser.f_appleKextOption))
158 al.append(al.makeFlagArg(args.parser.staticOption))
159 elif arg.opt is args.parser.f_indirectVirtualCallsOption:
160 al.append(al.makeFlagArg(args.parser.f_appleKextOption))
161 al.append(al.makeFlagArg(args.parser.staticOption))
162 elif arg.opt is args.parser.f_constantCfstringsOption:
Daniel Dunbar25d4a8f2009-01-13 21:07:43 +0000163 al.append(al.makeFlagArg(args.parser.m_constantCfstringsOption))
164 elif arg.opt is args.parser.f_noConstantCfstringsOption:
165 al.append(al.makeFlagArg(args.parser.m_noConstantCfstringsOption))
166 elif arg.opt is args.parser.WnonportableCfstringsOption:
167 al.append(al.makeFlagArg(args.parser.m_warnNonportableCfstringsOption))
168 elif arg.opt is args.parser.WnoNonportableCfstringsOption:
169 al.append(al.makeFlagArg(args.parser.m_noWarnNonportableCfstringsOption))
170 elif arg.opt is args.parser.f_pascalStringsOption:
171 al.append(al.makeFlagArg(args.parser.m_pascalStringsOption))
172 elif arg.opt is args.parser.f_noPascalStringsOption:
173 al.append(al.makeFlagArg(args.parser.m_noPascalStringsOption))
174 else:
175 al.append(arg)
176
177 # FIXME: Actually, gcc always adds this, but it is filtered
178 # for duplicates somewhere. This also changes the order of
179 # things, so look it up.
180 if arch and args.getValue(arch) == 'x86_64':
181 if not args.getLastArg(args.parser.m_64Option):
182 al.append(al.makeFlagArg(args.parser.m_64Option))
183
184 if not args.getLastArg(args.parser.m_tuneOption):
185 al.append(al.makeJoinedArg('core2',
186 args.parser.m_tuneOption))
187
188 return al
189
Daniel Dunbar43124722009-01-10 02:07:54 +0000190class Generic_GCC_ToolChain(ToolChain):
191 """Generic_GCC_ToolChain - A tool chain using the 'gcc' command to
192 perform all subcommands; this relies on gcc translating the
193 options appropriately."""
194
195 def __init__(self, driver):
196 super(Generic_GCC_ToolChain, self).__init__(driver)
197 self.toolMap = {
198 Phases.PreprocessPhase : Tools.GCC_PreprocessTool(),
Daniel Dunbarde388a52009-01-21 01:07:49 +0000199 Phases.AnalyzePhase : Tools.Clang_CompileTool(self),
200 Phases.SyntaxOnlyPhase : Tools.GCC_CompileTool(),
Daniel Dunbar43124722009-01-10 02:07:54 +0000201 Phases.CompilePhase : Tools.GCC_CompileTool(),
202 Phases.PrecompilePhase : Tools.GCC_PrecompileTool(),
203 Phases.AssemblePhase : Tools.GCC_AssembleTool(),
204 Phases.LinkPhase : Tools.GCC_LinkTool(),
205 }
206
207 def selectTool(self, action):
208 assert isinstance(action, Phases.JobAction)
209 return self.toolMap[action.phase.__class__]