blob: ac1cc0a0badae08f4fc23212418f2009a459ab20 [file] [log] [blame]
Daniel Dunbar11672ec2009-01-13 18:51:26 +00001import Arguments
Daniel Dunbar43124722009-01-10 02:07:54 +00002import Phases
3import Tools
Daniel Dunbar33a5d612009-01-14 01:03:36 +00004import Types
Daniel Dunbar43124722009-01-10 02:07:54 +00005
6###
7
8class ToolChain(object):
9 """ToolChain - Provide mappings of Actions to Tools."""
10
11 def __init__(self, driver):
12 self.driver = driver
13
14 def selectTool(self, action):
15 """selectTool - Return a Tool instance to use for handling
16 some particular action."""
17 abstract
18
Daniel Dunbar11672ec2009-01-13 18:51:26 +000019 def translateArgs(self, args, arch):
20 """translateArgs - Callback to allow argument translation for
21 an entire toolchain."""
22
23 # FIXME: Would be nice to move arch handling out of generic
24 # code.
25 if arch:
26 archName = args.getValue(arch)
27 al = Arguments.DerivedArgList(args)
28 for arg in args.args:
29 if arg.opt is args.parser.archOption:
30 if arg is arch:
31 al.append(arg)
32 elif arg.opt is args.parser.XarchOption:
33 if args.getJoinedValue(arg) == archName:
34 # FIXME: Fix this.
35 arg = args.parser.lookupOptForArg(Arguments.InputIndex(0, arg.index.pos + 1),
36 args.getSeparateValue(arg),
37 iter([]))
38 al.append(arg)
39 else:
40 al.append(arg)
41 return al
42 else:
43 return args
44
Daniel Dunbar9c257c32009-01-12 04:21:12 +000045class Darwin_X86_ToolChain(ToolChain):
Daniel Dunbar33a5d612009-01-14 01:03:36 +000046 def __init__(self, driver, darwinVersion, gccVersion, archName):
Daniel Dunbar9c257c32009-01-12 04:21:12 +000047 super(Darwin_X86_ToolChain, self).__init__(driver)
Daniel Dunbar9cb22532009-01-12 07:45:49 +000048 assert isinstance(darwinVersion, tuple) and len(darwinVersion) == 3
49 assert isinstance(gccVersion, tuple) and len(gccVersion) == 3
50 self.darwinVersion = darwinVersion
51 self.gccVersion = gccVersion
Daniel Dunbar33a5d612009-01-14 01:03:36 +000052 self.archName = archName
Daniel Dunbar9cb22532009-01-12 07:45:49 +000053
Daniel Dunbar43124722009-01-10 02:07:54 +000054 self.toolMap = {
Daniel Dunbar06172d62009-01-20 00:47:24 +000055 Phases.PreprocessPhase : Tools.Darwin_X86_PreprocessTool(self),
Daniel Dunbar11672ec2009-01-13 18:51:26 +000056 Phases.CompilePhase : Tools.Darwin_X86_CompileTool(self),
Daniel Dunbar4e3d7622009-01-20 05:51:52 +000057 Phases.PrecompilePhase : Tools.Darwin_X86_CompileTool(self),
Daniel Dunbar9cb22532009-01-12 07:45:49 +000058 Phases.AssemblePhase : Tools.Darwin_AssembleTool(self),
59 Phases.LinkPhase : Tools.Darwin_X86_LinkTool(self),
Daniel Dunbar43124722009-01-10 02:07:54 +000060 Phases.LipoPhase : Tools.LipoTool(),
61 }
Daniel Dunbar33a5d612009-01-14 01:03:36 +000062 self.clangTool = Tools.Clang_CompileTool()
Daniel Dunbar43124722009-01-10 02:07:54 +000063
Daniel Dunbar9cb22532009-01-12 07:45:49 +000064 def getToolChainDir(self):
65 return 'i686-apple-darwin%d/%s' % (self.darwinVersion[0],
66 '.'.join(map(str,self.gccVersion)))
67
68 def getProgramPath(self, name):
69 # FIXME: Implement proper search.
70 return '/usr/libexec/gcc/%s/%s' % (self.getToolChainDir(), name)
71
Daniel Dunbardff9f502009-01-12 18:51:02 +000072 def getMacosxVersionMin(self):
73 major,minor,minorminor = self.darwinVersion
74 return '%d.%d.%d' % (10, major-4, minor)
75
Daniel Dunbar43124722009-01-10 02:07:54 +000076 def selectTool(self, action):
77 assert isinstance(action, Phases.JobAction)
Daniel Dunbar33a5d612009-01-14 01:03:36 +000078
Daniel Dunbar89db21f2009-01-14 23:26:40 +000079 if self.driver.cccClang and self.archName == 'i386':
80 if (action.inputs[0].type in (Types.CType, Types.CTypeNoPP,
81 Types.ObjCType, Types.ObjCTypeNoPP) and
82 isinstance(action.phase, Phases.CompilePhase)):
83 return self.clangTool
84 elif (action.inputs[0].type in (Types.CHeaderType, Types.CHeaderNoPPType,
85 Types.ObjCHeaderType, Types.ObjCHeaderNoPPType) and
86 isinstance(action.phase, Phases.PrecompilePhase)):
87 return self.clangTool
Daniel Dunbar33a5d612009-01-14 01:03:36 +000088
Daniel Dunbar43124722009-01-10 02:07:54 +000089 return self.toolMap[action.phase.__class__]
90
Daniel Dunbar25d4a8f2009-01-13 21:07:43 +000091 def translateArgs(self, args, arch):
92 args = super(Darwin_X86_ToolChain, self).translateArgs(args, arch)
93
94 # If arch hasn't been bound we don't need to do anything yet.
95 if not arch:
96 return args
97
Daniel Dunbar9c3ea762009-01-16 20:25:36 +000098 # FIXME: We really want to get out of the tool chain level
99 # argument translation business, as it makes the driver
100 # functionality much more opaque. For now, we follow gcc
101 # closely solely for the purpose of easily achieving feature
102 # parity & testability. Once we have something that works, we
103 # should reevaluate each translation and try to push it down
104 # into tool specific logic.
105
Daniel Dunbar25d4a8f2009-01-13 21:07:43 +0000106 al = Arguments.DerivedArgList(args)
107 if not args.getLastArg(args.parser.m_macosxVersionMinOption):
108 al.append(al.makeJoinedArg(self.getMacosxVersionMin(),
109 args.parser.m_macosxVersionMinOption))
110 for arg in args:
Daniel Dunbar9c3ea762009-01-16 20:25:36 +0000111 # Sob. These is strictly gcc compatible for the time
112 # being. Apple gcc translates options twice, which means
113 # that self-expanding options add duplicates.
114 if arg.opt is args.parser.m_kernelOption:
115 al.append(arg)
116 al.append(al.makeFlagArg(args.parser.staticOption))
117 al.append(al.makeFlagArg(args.parser.staticOption))
118 elif arg.opt is args.parser.dependencyFileOption:
119 al.append(al.makeSeparateArg(args.getValue(arg),
120 args.parser.MFOption))
121 elif arg.opt is args.parser.gfullOption:
122 al.append(al.makeFlagArg(args.parser.gOption))
123 al.append(al.makeFlagArg(args.parser.f_noEliminateUnusedDebugSymbolsOption))
124 elif arg.opt is args.parser.gusedOption:
125 al.append(al.makeFlagArg(args.parser.gOption))
126 al.append(al.makeFlagArg(args.parser.f_eliminateUnusedDebugSymbolsOption))
127 elif arg.opt is args.parser.f_appleKextOption:
128 al.append(arg)
129 al.append(al.makeFlagArg(args.parser.staticOption))
130 al.append(al.makeFlagArg(args.parser.staticOption))
131 elif arg.opt is args.parser.f_terminatedVtablesOption:
132 al.append(al.makeFlagArg(args.parser.f_appleKextOption))
133 al.append(al.makeFlagArg(args.parser.staticOption))
134 elif arg.opt is args.parser.f_indirectVirtualCallsOption:
135 al.append(al.makeFlagArg(args.parser.f_appleKextOption))
136 al.append(al.makeFlagArg(args.parser.staticOption))
137 elif arg.opt is args.parser.f_constantCfstringsOption:
Daniel Dunbar25d4a8f2009-01-13 21:07:43 +0000138 al.append(al.makeFlagArg(args.parser.m_constantCfstringsOption))
139 elif arg.opt is args.parser.f_noConstantCfstringsOption:
140 al.append(al.makeFlagArg(args.parser.m_noConstantCfstringsOption))
141 elif arg.opt is args.parser.WnonportableCfstringsOption:
142 al.append(al.makeFlagArg(args.parser.m_warnNonportableCfstringsOption))
143 elif arg.opt is args.parser.WnoNonportableCfstringsOption:
144 al.append(al.makeFlagArg(args.parser.m_noWarnNonportableCfstringsOption))
145 elif arg.opt is args.parser.f_pascalStringsOption:
146 al.append(al.makeFlagArg(args.parser.m_pascalStringsOption))
147 elif arg.opt is args.parser.f_noPascalStringsOption:
148 al.append(al.makeFlagArg(args.parser.m_noPascalStringsOption))
149 else:
150 al.append(arg)
151
152 # FIXME: Actually, gcc always adds this, but it is filtered
153 # for duplicates somewhere. This also changes the order of
154 # things, so look it up.
155 if arch and args.getValue(arch) == 'x86_64':
156 if not args.getLastArg(args.parser.m_64Option):
157 al.append(al.makeFlagArg(args.parser.m_64Option))
158
159 if not args.getLastArg(args.parser.m_tuneOption):
160 al.append(al.makeJoinedArg('core2',
161 args.parser.m_tuneOption))
162
163 return al
164
Daniel Dunbar43124722009-01-10 02:07:54 +0000165class Generic_GCC_ToolChain(ToolChain):
166 """Generic_GCC_ToolChain - A tool chain using the 'gcc' command to
167 perform all subcommands; this relies on gcc translating the
168 options appropriately."""
169
170 def __init__(self, driver):
171 super(Generic_GCC_ToolChain, self).__init__(driver)
172 self.toolMap = {
173 Phases.PreprocessPhase : Tools.GCC_PreprocessTool(),
174 Phases.CompilePhase : Tools.GCC_CompileTool(),
175 Phases.PrecompilePhase : Tools.GCC_PrecompileTool(),
176 Phases.AssemblePhase : Tools.GCC_AssembleTool(),
177 Phases.LinkPhase : Tools.GCC_LinkTool(),
178 }
179
180 def selectTool(self, action):
181 assert isinstance(action, Phases.JobAction)
182 return self.toolMap[action.phase.__class__]