blob: 8511328512030550e1d53e60aa3a18b2a1820015 [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 = {
55 Phases.PreprocessPhase : Tools.GCC_PreprocessTool(),
Daniel Dunbar11672ec2009-01-13 18:51:26 +000056 Phases.CompilePhase : Tools.Darwin_X86_CompileTool(self),
Daniel Dunbar43124722009-01-10 02:07:54 +000057 Phases.PrecompilePhase : Tools.GCC_PrecompileTool(),
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
98 al = Arguments.DerivedArgList(args)
99 if not args.getLastArg(args.parser.m_macosxVersionMinOption):
100 al.append(al.makeJoinedArg(self.getMacosxVersionMin(),
101 args.parser.m_macosxVersionMinOption))
102 for arg in args:
103 if arg.opt is args.parser.f_constantCfstringsOption:
104 al.append(al.makeFlagArg(args.parser.m_constantCfstringsOption))
105 elif arg.opt is args.parser.f_noConstantCfstringsOption:
106 al.append(al.makeFlagArg(args.parser.m_noConstantCfstringsOption))
107 elif arg.opt is args.parser.WnonportableCfstringsOption:
108 al.append(al.makeFlagArg(args.parser.m_warnNonportableCfstringsOption))
109 elif arg.opt is args.parser.WnoNonportableCfstringsOption:
110 al.append(al.makeFlagArg(args.parser.m_noWarnNonportableCfstringsOption))
111 elif arg.opt is args.parser.f_pascalStringsOption:
112 al.append(al.makeFlagArg(args.parser.m_pascalStringsOption))
113 elif arg.opt is args.parser.f_noPascalStringsOption:
114 al.append(al.makeFlagArg(args.parser.m_noPascalStringsOption))
115 else:
116 al.append(arg)
117
118 # FIXME: Actually, gcc always adds this, but it is filtered
119 # for duplicates somewhere. This also changes the order of
120 # things, so look it up.
121 if arch and args.getValue(arch) == 'x86_64':
122 if not args.getLastArg(args.parser.m_64Option):
123 al.append(al.makeFlagArg(args.parser.m_64Option))
124
125 if not args.getLastArg(args.parser.m_tuneOption):
126 al.append(al.makeJoinedArg('core2',
127 args.parser.m_tuneOption))
128
129 return al
130
Daniel Dunbar43124722009-01-10 02:07:54 +0000131class Generic_GCC_ToolChain(ToolChain):
132 """Generic_GCC_ToolChain - A tool chain using the 'gcc' command to
133 perform all subcommands; this relies on gcc translating the
134 options appropriately."""
135
136 def __init__(self, driver):
137 super(Generic_GCC_ToolChain, self).__init__(driver)
138 self.toolMap = {
139 Phases.PreprocessPhase : Tools.GCC_PreprocessTool(),
140 Phases.CompilePhase : Tools.GCC_CompileTool(),
141 Phases.PrecompilePhase : Tools.GCC_PrecompileTool(),
142 Phases.AssemblePhase : Tools.GCC_AssembleTool(),
143 Phases.LinkPhase : Tools.GCC_LinkTool(),
144 }
145
146 def selectTool(self, action):
147 assert isinstance(action, Phases.JobAction)
148 return self.toolMap[action.phase.__class__]