# Copyright 2014-2021 The Meson development team
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
# http://www.apache.org/licenses/LICENSE-2.0
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
from . import backends
from . . import build
from . . import dependencies
from . . import mesonlib
from . . import mlog
import uuid , os , operator
import typing as T
from . . mesonlib import MesonException , OptionKey
from . . interpreter import Interpreter
INDENT = ' \t '
XCODETYPEMAP = { ' c ' : ' sourcecode.c.c ' ,
' a ' : ' archive.ar ' ,
' cc ' : ' sourcecode.cpp.cpp ' ,
' cxx ' : ' sourcecode.cpp.cpp ' ,
' cpp ' : ' sourcecode.cpp.cpp ' ,
' c++ ' : ' sourcecode.cpp.cpp ' ,
' m ' : ' sourcecode.c.objc ' ,
' mm ' : ' sourcecode.cpp.objcpp ' ,
' h ' : ' sourcecode.c.h ' ,
' hpp ' : ' sourcecode.cpp.h ' ,
' hxx ' : ' sourcecode.cpp.h ' ,
' hh ' : ' sourcecode.cpp.hh ' ,
' inc ' : ' sourcecode.c.h ' ,
' dylib ' : ' compiled.mach-o.dylib ' ,
' o ' : ' compiled.mach-o.objfile ' ,
' s ' : ' sourcecode.asm ' ,
' asm ' : ' sourcecode.asm ' ,
}
LANGNAMEMAP = { ' c ' : ' C ' ,
' cpp ' : ' CPLUSPLUS ' ,
' objc ' : ' OBJC ' ,
' objcpp ' : ' OBJCPLUSPLUS ' ,
}
OPT2XCODEOPT = { ' 0 ' : ' 0 ' ,
' g ' : ' 0 ' ,
' 1 ' : ' 1 ' ,
' 2 ' : ' 2 ' ,
' 3 ' : ' 3 ' ,
' s ' : ' s ' ,
}
BOOL2XCODEBOOL = { True : ' YES ' , False : ' NO ' }
class PbxItem :
def __init__ ( self , value , comment = ' ' ) :
self . value = value
self . comment = comment
class PbxArray :
def __init__ ( self ) :
self . items = [ ]
def add_item ( self , item , comment = ' ' ) :
if isinstance ( item , PbxArrayItem ) :
self . items . append ( item )
else :
self . items . append ( PbxArrayItem ( item , comment ) )
def write ( self , ofile , indent_level ) :
ofile . write ( ' ( \n ' )
indent_level + = 1
for i in self . items :
if i . comment :
ofile . write ( indent_level * INDENT + f ' { i . value } { i . comment } , \n ' )
else :
ofile . write ( indent_level * INDENT + f ' { i . value } , \n ' )
indent_level - = 1
ofile . write ( indent_level * INDENT + ' ); \n ' )
class PbxArrayItem :
def __init__ ( self , value , comment = ' ' ) :
self . value = value
if comment :
if ' /* ' in comment :
self . comment = comment
else :
self . comment = f ' /* { comment } */ '
else :
self . comment = comment
class PbxComment :
def __init__ ( self , text ) :
assert ( isinstance ( text , str ) )
assert ( ' /* ' not in text )
self . text = f ' /* { text } */ '
def write ( self , ofile , indent_level ) :
ofile . write ( f ' \n { self . text } \n ' )
class PbxDictItem :
def __init__ ( self , key , value , comment = ' ' ) :
self . key = key
self . value = value
if comment :
if ' /* ' in comment :
self . comment = comment
else :
self . comment = f ' /* { comment } */ '
else :
self . comment = comment
class PbxDict :
def __init__ ( self ) :
# This class is a bit weird, because we want to write PBX dicts in
# defined order _and_ we want to write intermediate comments also in order.
self . keys = set ( )
self . items = [ ]
def add_item ( self , key , value , comment = ' ' ) :
item = PbxDictItem ( key , value , comment )
assert ( key not in self . keys )
self . keys . add ( key )
self . items . append ( item )
def add_comment ( self , comment ) :
if isinstance ( comment , str ) :
self . items . append ( PbxComment ( str ) )
else :
assert ( isinstance ( comment , PbxComment ) )
self . items . append ( comment )
def write ( self , ofile , indent_level ) :
ofile . write ( ' { \n ' )
indent_level + = 1
for i in self . items :
if isinstance ( i , PbxComment ) :
i . write ( ofile , indent_level )
elif isinstance ( i , PbxDictItem ) :
if isinstance ( i . value , ( str , int ) ) :
if i . comment :
ofile . write ( indent_level * INDENT + f ' { i . key } = { i . value } { i . comment } ; \n ' )
else :
ofile . write ( indent_level * INDENT + f ' { i . key } = { i . value } ; \n ' )
elif isinstance ( i . value , PbxDict ) :
if i . comment :
ofile . write ( indent_level * INDENT + f ' { i . key } { i . comment } = ' )
else :
ofile . write ( indent_level * INDENT + f ' { i . key } = ' )
i . value . write ( ofile , indent_level )
elif isinstance ( i . value , PbxArray ) :
if i . comment :
ofile . write ( indent_level * INDENT + f ' { i . key } { i . comment } = ' )
else :
ofile . write ( indent_level * INDENT + f ' { i . key } = ' )
i . value . write ( ofile , indent_level )
else :
raise RuntimeError ( ' missing code ' )
else :
print ( i )
raise RuntimeError ( ' missing code2 ' )
indent_level - = 1
ofile . write ( indent_level * INDENT + ' } ' )
if indent_level == 0 :
ofile . write ( ' \n ' )
else :
ofile . write ( ' ; \n ' )
class XCodeBackend ( backends . Backend ) :
def __init__ ( self , build : T . Optional [ build . Build ] , interpreter : T . Optional [ Interpreter ] ) :
super ( ) . __init__ ( build , interpreter )
self . name = ' xcode '
self . project_uid = self . environment . coredata . lang_guids [ ' default ' ] . replace ( ' - ' , ' ' ) [ : 24 ]
self . buildtype = self . environment . coredata . get_option ( OptionKey ( ' buildtype ' ) )
self . project_conflist = self . gen_id ( )
self . maingroup_id = self . gen_id ( )
self . all_id = self . gen_id ( )
self . all_buildconf_id = self . gen_id ( )
self . buildtypes = [ self . buildtype ]
self . test_id = self . gen_id ( )
self . test_command_id = self . gen_id ( )
self . test_buildconf_id = self . gen_id ( )
self . top_level_dict = PbxDict ( )
def write_pbxfile ( self , top_level_dict , ofilename ) :
with open ( ofilename , ' w ' ) as ofile :
ofile . write ( ' // !$*UTF8*$! \n ' )
top_level_dict . write ( ofile , 0 )
def gen_id ( self ) :
return str ( uuid . uuid4 ( ) ) . upper ( ) . replace ( ' - ' , ' ' ) [ : 24 ]
def get_target_dir ( self , target ) :
dirname = os . path . join ( target . get_subdir ( ) , self . environment . coredata . get_option ( OptionKey ( ' buildtype ' ) ) )
os . makedirs ( os . path . join ( self . environment . get_build_dir ( ) , dirname ) , exist_ok = True )
return dirname
def target_to_build_root ( self , target ) :
if self . get_target_dir ( target ) == ' ' :
return ' '
directories = os . path . normpath ( self . get_target_dir ( target ) ) . split ( os . sep )
return os . sep . join ( [ ' .. ' ] * len ( directories ) )
def generate ( self ) :
test_data = self . serialize_tests ( ) [ 0 ]
self . generate_filemap ( )
self . generate_buildmap ( )
self . generate_buildstylemap ( )
self . generate_build_phase_map ( )
self . generate_build_configuration_map ( )
self . generate_build_configurationlist_map ( )
self . generate_project_configurations_map ( )
self . generate_buildall_configurations_map ( )
self . generate_test_configurations_map ( )
self . generate_native_target_map ( )
self . generate_native_frameworks_map ( )
self . generate_source_phase_map ( )
self . generate_target_dependency_map ( )
self . generate_pbxdep_map ( )
self . generate_containerproxy_map ( )
self . proj_dir = os . path . join ( self . environment . get_build_dir ( ) , self . build . project_name + ' .xcodeproj ' )
os . makedirs ( self . proj_dir , exist_ok = True )
self . proj_file = os . path . join ( self . proj_dir , ' project.pbxproj ' )
objects_dict = self . generate_prefix ( self . top_level_dict )
objects_dict . add_comment ( PbxComment ( ' Begin PBXAggregateTarget section ' ) )
self . generate_pbx_aggregate_target ( objects_dict )
objects_dict . add_comment ( PbxComment ( ' End PBXAggregateTarget section ' ) )
objects_dict . add_comment ( PbxComment ( ' Begin PBXBuildFile section ' ) )
self . generate_pbx_build_file ( objects_dict )
objects_dict . add_comment ( PbxComment ( ' End PBXBuildFile section ' ) )
objects_dict . add_comment ( PbxComment ( ' Begin PBXBuildStyle section ' ) )
self . generate_pbx_build_style ( objects_dict )
objects_dict . add_comment ( PbxComment ( ' End PBXBuildStyle section ' ) )
objects_dict . add_comment ( PbxComment ( ' Begin PBXContainerItemProxy section ' ) )
self . generate_pbx_container_item_proxy ( objects_dict )
objects_dict . add_comment ( PbxComment ( ' End PBXContainerItemProxy section ' ) )
objects_dict . add_comment ( PbxComment ( ' Begin PBXFileReference section ' ) )
self . generate_pbx_file_reference ( objects_dict )
objects_dict . add_comment ( PbxComment ( ' End PBXFileReference section ' ) )
objects_dict . add_comment ( PbxComment ( ' Begin PBXFrameworksBuildPhase section ' ) )
self . generate_pbx_frameworks_buildphase ( objects_dict )
objects_dict . add_comment ( PbxComment ( ' End PBXFrameworksBuildPhase section ' ) )
objects_dict . add_comment ( PbxComment ( ' Begin PBXGroup section ' ) )
self . generate_pbx_group ( objects_dict )
objects_dict . add_comment ( PbxComment ( ' End PBXGroup section ' ) )
objects_dict . add_comment ( PbxComment ( ' Begin PBXNativeTarget section ' ) )
self . generate_pbx_native_target ( objects_dict )
objects_dict . add_comment ( PbxComment ( ' End PBXNativeTarget section ' ) )
objects_dict . add_comment ( PbxComment ( ' Begin PBXProject section ' ) )
self . generate_pbx_project ( objects_dict )
objects_dict . add_comment ( PbxComment ( ' End PBXProject section ' ) )
objects_dict . add_comment ( PbxComment ( ' Begin PBXShellScriptBuildPhase section ' ) )
self . generate_pbx_shell_build_phase ( objects_dict , test_data )
objects_dict . add_comment ( PbxComment ( ' End PBXShellScriptBuildPhase section ' ) )
objects_dict . add_comment ( PbxComment ( ' Begin PBXSourcesBuildPhase section ' ) )
self . generate_pbx_sources_build_phase ( objects_dict )
objects_dict . add_comment ( PbxComment ( ' End PBXSourcesBuildPhase section ' ) )
objects_dict . add_comment ( PbxComment ( ' Begin PBXTargetDependency section ' ) )
self . generate_pbx_target_dependency ( objects_dict )
objects_dict . add_comment ( PbxComment ( ' End PBXTargetDependency section ' ) )
objects_dict . add_comment ( PbxComment ( ' Begin XCBuildConfiguration section ' ) )
self . generate_xc_build_configuration ( objects_dict )
objects_dict . add_comment ( PbxComment ( ' End XCBuildConfiguration section ' ) )
objects_dict . add_comment ( PbxComment ( ' Begin XCConfigurationList section ' ) )
self . generate_xc_configurationList ( objects_dict )
objects_dict . add_comment ( PbxComment ( ' End XCConfigurationList section ' ) )
self . generate_suffix ( self . top_level_dict )
self . write_pbxfile ( self . top_level_dict , self . proj_file )
def get_xcodetype ( self , fname ) :
xcodetype = XCODETYPEMAP . get ( fname . split ( ' . ' ) [ - 1 ] . lower ( ) )
if not xcodetype :
xcodetype = ' sourcecode.unknown '
mlog . warning ( f ' Unknown file type " { fname } " fallbacking to " { xcodetype } " . Xcode project might be malformed. ' )
return xcodetype
def generate_filemap ( self ) :
self . filemap = { } # Key is source file relative to src root.
self . target_filemap = { }
for name , t in self . build . get_build_targets ( ) . items ( ) :
for s in t . sources :
if isinstance ( s , mesonlib . File ) :
s = os . path . join ( s . subdir , s . fname )
self . filemap [ s ] = self . gen_id ( )
for o in t . objects :
if isinstance ( o , str ) :
o = os . path . join ( t . subdir , o )
self . filemap [ o ] = self . gen_id ( )
self . target_filemap [ name ] = self . gen_id ( )
def generate_buildmap ( self ) :
self . buildmap = { }
for t in self . build . get_build_targets ( ) . values ( ) :
for s in t . sources :
s = os . path . join ( s . subdir , s . fname )
self . buildmap [ s ] = self . gen_id ( )
for o in t . objects :
o = os . path . join ( t . subdir , o )
if isinstance ( o , str ) :
self . buildmap [ o ] = self . gen_id ( )
def generate_buildstylemap ( self ) :
self . buildstylemap = { self . buildtype : self . gen_id ( ) }
def generate_build_phase_map ( self ) :
for tname , t in self . build . get_build_targets ( ) . items ( ) :
# generate id for our own target-name
t . buildphasemap = { }
t . buildphasemap [ tname ] = self . gen_id ( )
# each target can have it's own Frameworks/Sources/..., generate id's for those
t . buildphasemap [ ' Frameworks ' ] = self . gen_id ( )
t . buildphasemap [ ' Resources ' ] = self . gen_id ( )
t . buildphasemap [ ' Sources ' ] = self . gen_id ( )
def generate_build_configuration_map ( self ) :
self . buildconfmap = { }
for t in self . build . get_build_targets ( ) :
bconfs = { self . buildtype : self . gen_id ( ) }
self . buildconfmap [ t ] = bconfs
def generate_project_configurations_map ( self ) :
self . project_configurations = { self . buildtype : self . gen_id ( ) }
def generate_buildall_configurations_map ( self ) :
self . buildall_configurations = { self . buildtype : self . gen_id ( ) }
def generate_test_configurations_map ( self ) :
self . test_configurations = { self . buildtype : self . gen_id ( ) }
def generate_build_configurationlist_map ( self ) :
self . buildconflistmap = { }
for t in self . build . get_build_targets ( ) :
self . buildconflistmap [ t ] = self . gen_id ( )
def generate_native_target_map ( self ) :
self . native_targets = { }
for t in self . build . get_build_targets ( ) :
self . native_targets [ t ] = self . gen_id ( )
def generate_native_frameworks_map ( self ) :
self . native_frameworks = { }
self . native_frameworks_fileref = { }
for t in self . build . get_build_targets ( ) . values ( ) :
for dep in t . get_external_deps ( ) :
if isinstance ( dep , dependencies . AppleFrameworks ) :
for f in dep . frameworks :
self . native_frameworks [ f ] = self . gen_id ( )
self . native_frameworks_fileref [ f ] = self . gen_id ( )
def generate_target_dependency_map ( self ) :
self . target_dependency_map = { }
for tname , t in self . build . get_build_targets ( ) . items ( ) :
for target in t . link_targets :
self . target_dependency_map [ ( tname , target . get_basename ( ) ) ] = self . gen_id ( )
def generate_pbxdep_map ( self ) :
self . pbx_dep_map = { }
for t in self . build . get_build_targets ( ) :
self . pbx_dep_map [ t ] = self . gen_id ( )
def generate_containerproxy_map ( self ) :
self . containerproxy_map = { }
for t in self . build . get_build_targets ( ) :
self . containerproxy_map [ t ] = self . gen_id ( )
def generate_source_phase_map ( self ) :
self . source_phase = { }
for t in self . build . get_build_targets ( ) :
self . source_phase [ t ] = self . gen_id ( )
def generate_pbx_aggregate_target ( self , objects_dict ) :
target_dependencies = list ( map ( lambda t : self . pbx_dep_map [ t ] , self . build . get_build_targets ( ) ) )
aggregated_targets = [ ]
aggregated_targets . append ( ( self . all_id , ' ALL_BUILD ' , self . all_buildconf_id , [ ] , target_dependencies ) )
aggregated_targets . append ( ( self . test_id , ' RUN_TESTS ' , self . test_buildconf_id , [ self . test_command_id ] , [ ] ) )
# Sort objects by ID before writing
sorted_aggregated_targets = sorted ( aggregated_targets , key = operator . itemgetter ( 0 ) )
for t in sorted_aggregated_targets :
agt_dict = PbxDict ( )
name = t [ 1 ]
buildconf_id = t [ 2 ]
build_phases = t [ 3 ]
dependencies = t [ 4 ]
agt_dict . add_item ( ' isa ' , ' PBXAggregateTarget ' )
agt_dict . add_item ( ' buildConfigurationList ' , buildconf_id , f ' Build configuration list for PBXAggregateTarget " { name } " ' )
bp_arr = PbxArray ( )
agt_dict . add_item ( ' buildPhases ' , bp_arr )
for bp in build_phases :
bp_arr . add_item ( bp , ' ShellScript ' )
dep_arr = PbxArray ( )
agt_dict . add_item ( ' dependencies ' , dep_arr )
for td in dependencies :
dep_arr . add_item ( td , ' PBXTargetDependency ' )
agt_dict . add_item ( ' name ' , name )
agt_dict . add_item ( ' productName ' , name )
objects_dict . add_item ( t [ 0 ] , agt_dict , name )
def generate_pbx_build_file ( self , objects_dict ) :
for t in self . build . get_build_targets ( ) . values ( ) :
for dep in t . get_external_deps ( ) :
if isinstance ( dep , dependencies . AppleFrameworks ) :
for f in dep . frameworks :
fw_dict = PbxDict ( )
objects_dict . add : item ( self . native_frameworks [ f ] , fw_dict , f ' { f } .framework in Frameworks ' )
fw_dict . add_item ( ' isa ' , ' PBXBuildFile ' )
fw_dict . add_item ( ' fileRef ' , self . native_frameworks_fileref [ f ] , f )
for s in t . sources :
if isinstance ( s , mesonlib . File ) :
s = os . path . join ( s . subdir , s . fname )
if isinstance ( s , str ) :
s = os . path . join ( t . subdir , s )
sdict = PbxDict ( )
idval = self . buildmap [ s ]
fullpath = os . path . join ( self . environment . get_source_dir ( ) , s )
fileref = self . filemap [ s ]
fullpath2 = fullpath
compiler_args = ' '
sdict . add_item ( ' isa ' , ' PBXBuildFile ' )
sdict . add_item ( ' fileRef ' , fileref , fullpath2 )
settingdict = PbxDict ( )
settingdict . add_item ( ' COMPILER_FLAGS ' , ' " ' + compiler_args + ' " ' )
sdict . add_item ( ' settings ' , settingdict )
objects_dict . add_item ( idval , sdict )
for o in t . objects :
o = os . path . join ( t . subdir , o )
idval = self . buildmap [ o ]
fileref = self . filemap [ o ]
fullpath = os . path . join ( self . environment . get_source_dir ( ) , o )
fullpath2 = fullpath
o_dict = PbxDict ( )
objects_dict . add_item ( idval , o_dict , fullpath )
o_dict . add_item ( ' isa ' , ' PBXBuildFile ' )
o_dict . add_item ( ' fileRef ' , fileref , fullpath2 )
def generate_pbx_build_style ( self , objects_dict ) :
# FIXME: Xcode 9 and later does not uses PBXBuildStyle and it gets removed. Maybe we can remove this part.
for name , idval in self . buildstylemap . items ( ) :
styledict = PbxDict ( )
objects_dict . add_item ( idval , styledict , name )
styledict . add_item ( ' isa ' , ' PBXBuildStyle ' )
settings_dict = PbxDict ( )
styledict . add_item ( ' buildSettings ' , settings_dict )
settings_dict . add_item ( ' COPY_PHASE_STRIP ' , ' NO ' )
styledict . add_item ( ' name ' , f ' " { name } " ' )
def generate_pbx_container_item_proxy ( self , objects_dict ) :
for t in self . build . get_build_targets ( ) :
proxy_dict = PbxDict ( )
objects_dict . add_item ( self . containerproxy_map [ t ] , proxy_dict , ' PBXContainerItemProxy ' )
proxy_dict . add_item ( ' isa ' , ' PBXContainerItemProxy ' )
proxy_dict . add_item ( ' containerPortal ' , self . project_uid , ' Project object ' )
proxy_dict . add_item ( ' proxyType ' , ' 1 ' )
proxy_dict . add_item ( ' remoteGlobalIDString ' , self . native_targets [ t ] )
proxy_dict . add_item ( ' remoteInfo ' , ' " ' + t + ' " ' )
def generate_pbx_file_reference ( self , objects_dict ) :
for t in self . build . get_build_targets ( ) . values ( ) :
for dep in t . get_external_deps ( ) :
if isinstance ( dep , dependencies . AppleFrameworks ) :
for f in dep . frameworks :
# FIXME not ported
fw_dict = PbxDict ( )
objects_dict . add_item ( self . native_frameworks_fileref [ f ] , fw_dict , f )
fw_dict . add_item ( ' isa ' , ' PBXFileReference ' )
fw_dict . add_item ( ' lastKnownFileType ' , ' wrapper.framework ' )
fw_dict . add_item ( ' name ' , f ' { f } .framework ' )
fw_dict . add_item ( ' path ' , f ' System/Library/Frameworks/ { f } .framework ' )
fw_dict . add_item ( ' sourceTree ' , ' SDKROOT ' )
for fname , idval in self . filemap . items ( ) :
src_dict = PbxDict ( )
fullpath = os . path . join ( self . environment . get_source_dir ( ) , fname )
xcodetype = self . get_xcodetype ( fname )
name = os . path . basename ( fname )
path = fname
objects_dict . add_item ( idval , src_dict , fullpath )
src_dict . add_item ( ' isa ' , ' PBXFileReference ' )
src_dict . add_item ( ' explicitFileType ' , ' " ' + xcodetype + ' " ' )
src_dict . add_item ( ' fileEncoding ' , ' 4 ' )
src_dict . add_item ( ' name ' , ' " ' + name + ' " ' )
src_dict . add_item ( ' path ' , ' " ' + path + ' " ' )
src_dict . add_item ( ' sourceTree ' , ' SOURCE_ROOT ' )
for tname , idval in self . target_filemap . items ( ) :
target_dict = PbxDict ( )
objects_dict . add_item ( idval , target_dict , tname )
t = self . build . get_build_targets ( ) [ tname ]
fname = t . get_filename ( )
reftype = 0
if isinstance ( t , build . Executable ) :
typestr = ' compiled.mach-o.executable '
path = fname
elif isinstance ( t , build . SharedLibrary ) :
typestr = self . get_xcodetype ( ' dummy.dylib ' )
path = fname
else :
typestr = self . get_xcodetype ( fname )
path = ' " %s " ' % t . get_filename ( )
target_dict . add_item ( ' isa ' , ' PBXFileReference ' )
target_dict . add_item ( ' explicitFileType ' , ' " ' + typestr + ' " ' )
target_dict . add_item ( ' path ' , path )
target_dict . add_item ( ' refType ' , reftype )
target_dict . add_item ( ' sourceTree ' , ' BUILT_PRODUCTS_DIR ' )
def generate_pbx_frameworks_buildphase ( self , objects_dict ) :
for t in self . build . get_build_targets ( ) . values ( ) :
bt_dict = PbxDict ( )
objects_dict . add_item ( t . buildphasemap [ ' Frameworks ' ] , bt_dict , ' Frameworks ' )
bt_dict . add_item ( ' isa ' , ' PBXFrameworksBuildPhase ' )
bt_dict . add_item ( ' buildActionMask ' , 2147483647 )
file_list = PbxArray ( )
bt_dict . add_item ( ' files ' , file_list )
for dep in t . get_external_deps ( ) :
if isinstance ( dep , dependencies . AppleFrameworks ) :
for f in dep . frameworks :
file_list . add_item ( self . native_frameworks [ f ] , f ' { f } .framework in Frameworks ' )
bt_dict . add_item ( ' runOnlyForDeploymentPostprocessing ' , 0 )
def generate_pbx_group ( self , objects_dict ) :
groupmap = { }
target_src_map = { }
for t in self . build . get_build_targets ( ) :
groupmap [ t ] = self . gen_id ( )
target_src_map [ t ] = self . gen_id ( )
sources_id = self . gen_id ( )
resources_id = self . gen_id ( )
products_id = self . gen_id ( )
frameworks_id = self . gen_id ( )
main_dict = PbxDict ( )
objects_dict . add_item ( self . maingroup_id , main_dict )
main_dict . add_item ( ' isa ' , ' PBXGroup ' )
main_children = PbxArray ( )
main_dict . add_item ( ' children ' , main_children )
main_children . add_item ( sources_id , ' Sources ' )
main_children . add_item ( resources_id , ' Resources ' )
main_children . add_item ( products_id , ' Products ' )
main_children . add_item ( frameworks_id , ' Frameworks ' )
main_dict . add_item ( ' sourceTree ' , ' " <group> " ' )
# Sources
source_dict = PbxDict ( )
objects_dict . add_item ( sources_id , source_dict , ' Sources ' )
source_dict . add_item ( ' isa ' , ' PBXGroup ' )
source_children = PbxArray ( )
source_dict . add_item ( ' children ' , source_children )
for t in self . build . get_build_targets ( ) :
source_children . add_item ( groupmap [ t ] , t )
source_dict . add_item ( ' name ' , ' Sources ' )
source_dict . add_item ( ' sourceTree ' , ' " <group> " ' )
resource_dict = PbxDict ( )
objects_dict . add_item ( resources_id , resource_dict , ' Resources ' )
resource_dict . add_item ( ' isa ' , ' PBXGroup ' )
resource_children = PbxArray ( )
resource_dict . add_item ( ' children ' , resource_children )
resource_dict . add_item ( ' name ' , ' Resources ' )
resource_dict . add_item ( ' sourceTree ' , ' " <group> " ' )
frameworks_dict = PbxDict ( )
objects_dict . add_item ( frameworks_id , frameworks_dict , ' Frameworks ' )
frameworks_dict . add_item ( ' isa ' , ' PBXGroup ' )
frameworks_children = PbxArray ( )
frameworks_dict . add_item ( ' children ' , frameworks_children )
# write frameworks
for t in self . build . get_build_targets ( ) . values ( ) :
for dep in t . get_external_deps ( ) :
if isinstance ( dep , dependencies . AppleFrameworks ) :
for f in dep . frameworks :
frameworks_children . add_item ( self . native_frameworks_fileref [ f ] , f )
frameworks_dict . add_item ( ' name ' , ' Frameworks ' )
frameworks_dict . add_item ( ' sourceTree ' , ' " <group> " ' )
# Targets
for t in self . build . get_build_targets ( ) :
target_dict = PbxDict ( )
objects_dict . add_item ( groupmap [ t ] , target_dict , t )
target_dict . add_item ( ' isa ' , ' PBXGroup ' )
target_children = PbxArray ( )
target_dict . add_item ( ' children ' , target_children )
target_children . add_item ( target_src_map [ t ] , ' Source files ' )
target_dict . add_item ( ' name ' , f ' " { t } " ' )
target_dict . add_item ( ' sourceTree ' , ' " <group> " ' )
source_files_dict = PbxDict ( )
objects_dict . add_item ( target_src_map [ t ] , source_files_dict , ' Source files ' )
source_files_dict . add_item ( ' isa ' , ' PBXGroup ' )
source_file_children = PbxArray ( )
source_files_dict . add_item ( ' children ' , source_file_children )
for s in self . build . get_build_targets ( ) [ t ] . sources :
s = os . path . join ( s . subdir , s . fname )
if isinstance ( s , str ) :
source_file_children . add_item ( self . filemap [ s ] , s )
for o in self . build . get_build_targets ( ) [ t ] . objects :
o = os . path . join ( self . build . get_build_targets ( ) [ t ] . subdir , o )
source_file_children . add_item ( self . filemap [ o ] , o )
source_files_dict . add_item ( ' name ' , ' " Source files " ' )
source_files_dict . add_item ( ' sourceTree ' , ' " <group> " ' )
# And finally products
product_dict = PbxDict ( )
objects_dict . add_item ( products_id , product_dict , ' Products ' )
product_dict . add_item ( ' isa ' , ' PBXGroup ' )
product_children = PbxArray ( )
product_dict . add_item ( ' children ' , product_children )
for t in self . build . get_build_targets ( ) :
product_children . add_item ( self . target_filemap [ t ] , t )
product_dict . add_item ( ' name ' , ' Products ' )
product_dict . add_item ( ' sourceTree ' , ' " <group> " ' )
def generate_pbx_native_target ( self , objects_dict ) :
for tname , idval in self . native_targets . items ( ) :
ntarget_dict = PbxDict ( )
t = self . build . get_build_targets ( ) [ tname ]
objects_dict . add_item ( idval , ntarget_dict , tname )
ntarget_dict . add_item ( ' isa ' , ' PBXNativeTarget ' )
ntarget_dict . add_item ( ' buildConfigurationList ' , self . buildconflistmap [ tname ] , f ' Build configuration list for PBXNativeTarget " { tname } " ' )
buildphases_array = PbxArray ( )
ntarget_dict . add_item ( ' buildPhases ' , buildphases_array )
for bpname , bpval in t . buildphasemap . items ( ) :
buildphases_array . add_item ( bpval , f ' { bpname } yyy ' )
ntarget_dict . add_item ( ' buildRules ' , PbxArray ( ) )
dep_array = PbxArray ( )
ntarget_dict . add_item ( ' dependencies ' , dep_array )
for lt in self . build . get_build_targets ( ) [ tname ] . link_targets :
# NOT DOCUMENTED, may need to make different links
# to same target have different targetdependency item.
idval = self . pbx_dep_map [ lt . get_id ( ) ]
dep_array . add_item ( idval , ' PBXTargetDependency ' )
ntarget_dict . add_item ( ' name ' , f ' " { tname } " ' )
ntarget_dict . add_item ( ' productName ' , f ' " { tname } " ' )
ntarget_dict . add_item ( ' productReference ' , self . target_filemap [ tname ] , tname )
if isinstance ( t , build . Executable ) :
typestr = ' com.apple.product-type.tool '
elif isinstance ( t , build . StaticLibrary ) :
typestr = ' com.apple.product-type.library.static '
elif isinstance ( t , build . SharedLibrary ) :
typestr = ' com.apple.product-type.library.dynamic '
else :
raise MesonException ( ' Unknown target type for %s ' % tname )
ntarget_dict . add_item ( ' productType ' , f ' " { typestr } " ' )
def generate_pbx_project ( self , objects_dict ) :
project_dict = PbxDict ( )
objects_dict . add_item ( self . project_uid , project_dict , ' Project object ' )
project_dict . add_item ( ' isa ' , ' PBXProject ' )
attr_dict = PbxDict ( )
project_dict . add_item ( ' attributes ' , attr_dict )
attr_dict . add_item ( ' BuildIndependentTargetsInParallel ' , ' YES ' )
project_dict . add_item ( ' buildConfigurationList ' , self . project_conflist , f ' Build configuration list for PBXProject " { self . build . project_name } " ' )
project_dict . add_item ( ' buildSettings ' , PbxDict ( ) )
style_arr = PbxArray ( )
project_dict . add_item ( ' buildStyles ' , style_arr )
for name , idval in self . buildstylemap . items ( ) :
style_arr . add_item ( idval , name )
project_dict . add_item ( ' compatibilityVersion ' , ' " Xcode 3.2 " ' )
project_dict . add_item ( ' hasScannedForEncodings ' , 0 )
project_dict . add_item ( ' mainGroup ' , self . maingroup_id )
project_dict . add_item ( ' projectDirPath ' , f ' " { self . build_to_src } " ' )
project_dict . add_item ( ' projectRoot ' , ' " " ' )
targets_arr = PbxArray ( )
project_dict . add_item ( ' targets ' , targets_arr )
targets_arr . add_item ( self . all_id , ' ALL_BUILD ' )
targets_arr . add_item ( self . test_id , ' RUN_TESTS ' )
for t in self . build . get_build_targets ( ) :
targets_arr . add_item ( self . native_targets [ t ] , t )
def generate_pbx_shell_build_phase ( self , objects_dict , test_data ) :
shell_dict = PbxDict ( )
objects_dict . add_item ( self . test_command_id , shell_dict , ' ShellScript ' )
shell_dict . add_item ( ' isa ' , ' PBXShellScriptBuildPhase ' )
shell_dict . add_item ( ' buildActionMask ' , 2147483647 )
shell_dict . add_item ( ' files ' , PbxArray ( ) )
shell_dict . add_item ( ' inputPaths ' , PbxArray ( ) )
shell_dict . add_item ( ' outputPaths ' , PbxArray ( ) )
shell_dict . add_item ( ' runOnlyForDeploymentPostprocessing ' , 0 )
shell_dict . add_item ( ' shellPath ' , ' /bin/sh ' )
cmd = mesonlib . get_meson_command ( ) + [ ' test ' , ' --no-rebuild ' , ' -C ' , self . environment . get_build_dir ( ) ]
cmdstr = ' ' . join ( [ " ' %s ' " % i for i in cmd ] )
shell_dict . add_item ( ' shellScript ' , f ' " { cmdstr } " ' )
shell_dict . add_item ( ' showEnvVarsInLog ' , 0 )
def generate_pbx_sources_build_phase ( self , objects_dict ) :
for name in self . source_phase . keys ( ) :
phase_dict = PbxDict ( )
t = self . build . get_build_targets ( ) [ name ]
objects_dict . add_item ( t . buildphasemap [ name ] , phase_dict , ' Sources ' )
phase_dict . add_item ( ' isa ' , ' PBXSourcesBuildPhase ' )
phase_dict . add_item ( ' buildActionMask ' , 2147483647 )
file_arr = PbxArray ( )
phase_dict . add_item ( ' files ' , file_arr )
for s in self . build . get_build_targets ( ) [ name ] . sources :
s = os . path . join ( s . subdir , s . fname )
if not self . environment . is_header ( s ) :
file_arr . add_item ( self . buildmap [ s ] , os . path . join ( self . environment . get_source_dir ( ) , s ) )
phase_dict . add_item ( ' runOnlyForDeploymentPostprocessing ' , 0 )
def generate_pbx_target_dependency ( self , objects_dict ) :
targets = [ ]
for t in self . build . get_build_targets ( ) :
idval = self . pbx_dep_map [ t ] # VERIFY: is this correct?
targets . append ( ( idval , self . native_targets [ t ] , t , self . containerproxy_map [ t ] ) )
# Sort object by ID
sorted_targets = sorted ( targets , key = operator . itemgetter ( 0 ) )
for t in sorted_targets :
t_dict = PbxDict ( )
objects_dict . add_item ( t [ 0 ] , t_dict , ' PBXTargetDependency ' )
t_dict . add_item ( ' isa ' , ' PBXTargetDependency ' )
t_dict . add_item ( ' target ' , t [ 1 ] , t [ 2 ] )
t_dict . add_item ( ' targetProxy ' , t [ 3 ] , ' PBXContainerItemProxy ' )
def generate_xc_build_configuration ( self , objects_dict ) :
# First the setup for the toplevel project.
for buildtype in self . buildtypes :
bt_dict = PbxDict ( )
objects_dict . add_item ( self . project_configurations [ buildtype ] , bt_dict , buildtype )
bt_dict . add_item ( ' isa ' , ' XCBuildConfiguration ' )
settings_dict = PbxDict ( )
bt_dict . add_item ( ' buildSettings ' , settings_dict )
settings_dict . add_item ( ' ARCHS ' , ' " $(NATIVE_ARCH_ACTUAL) " ' )
settings_dict . add_item ( ' ONLY_ACTIVE_ARCH ' , ' YES ' )
settings_dict . add_item ( ' SDKROOT ' , ' " macosx " ' )
settings_dict . add_item ( ' SYMROOT ' , ' " %s /build " ' % self . environment . get_build_dir ( ) )
bt_dict . add_item ( ' name ' , f ' " { buildtype } " ' )
# Then the all target.
for buildtype in self . buildtypes :
bt_dict = PbxDict ( )
objects_dict . add_item ( self . buildall_configurations [ buildtype ] , bt_dict , buildtype )
bt_dict . add_item ( ' isa ' , ' XCBuildConfiguration ' )
settings_dict = PbxDict ( )
bt_dict . add_item ( ' buildSettings ' , settings_dict )
settings_dict . add_item ( ' COMBINE_HIDPI_IMAGES ' , ' YES ' )
settings_dict . add_item ( ' GCC_INLINES_ARE_PRIVATE_EXTERN ' , ' NO ' )
settings_dict . add_item ( ' GCC_PREPROCESSOR_DEFINITIONS ' , ' " " ' )
settings_dict . add_item ( ' GCC_SYMBOLS_PRIVATE_EXTERN ' , ' NO ' )
settings_dict . add_item ( ' INSTALL_PATH ' , ' " " ' )
settings_dict . add_item ( ' OTHER_CFLAGS ' , ' " " ' )
settings_dict . add_item ( ' OTHER_LDFLAGS ' , ' " " ' )
settings_dict . add_item ( ' OTHER_REZFLAGS ' , ' " " ' )
settings_dict . add_item ( ' PRODUCT_NAME ' , ' ALL_BUILD ' )
settings_dict . add_item ( ' SECTORDER_FLAGS ' , ' " " ' )
settings_dict . add_item ( ' SYMROOT ' , ' " %s " ' % self . environment . get_build_dir ( ) )
settings_dict . add_item ( ' USE_HEADERMAP ' , ' NO ' )
warn_array = PbxArray ( )
settings_dict . add_item ( ' WARNING_CFLAGS ' , warn_array )
warn_array . add_item ( ' " -Wmost " ' )
warn_array . add_item ( ' " -Wno-four-char-constants " ' )
warn_array . add_item ( ' " -Wno-unknown-pragmas " ' )
bt_dict . add_item ( ' name ' , f ' " { buildtype } " ' )
# Then the test target.
for buildtype in self . buildtypes :
bt_dict = PbxDict ( )
objects_dict . add_item ( self . test_configurations [ buildtype ] , bt_dict , buildtype )
bt_dict . add_item ( ' isa ' , ' XCBuildConfiguration ' )
settings_dict = PbxDict ( )
bt_dict . add_item ( ' buildSettings ' , settings_dict )
settings_dict . add_item ( ' COMBINE_HIDPI_IMAGES ' , ' YES ' )
settings_dict . add_item ( ' GCC_INLINES_ARE_PRIVATE_EXTERN ' , ' NO ' )
settings_dict . add_item ( ' GCC_PREPROCESSOR_DEFINITIONS ' , ' " " ' )
settings_dict . add_item ( ' GCC_SYMBOLS_PRIVATE_EXTERN ' , ' NO ' )
settings_dict . add_item ( ' INSTALL_PATH ' , ' " " ' )
settings_dict . add_item ( ' OTHER_CFLAGS ' , ' " " ' )
settings_dict . add_item ( ' OTHER_LDFLAGS ' , ' " " ' )
settings_dict . add_item ( ' OTHER_REZFLAGS ' , ' " " ' )
settings_dict . add_item ( ' PRODUCT_NAME ' , ' RUN_TESTS ' )
settings_dict . add_item ( ' SECTORDER_FLAGS ' , ' " " ' )
settings_dict . add_item ( ' SYMROOT ' , ' " %s " ' % self . environment . get_build_dir ( ) )
settings_dict . add_item ( ' USE_HEADERMAP ' , ' NO ' )
warn_array = PbxArray ( )
settings_dict . add_item ( ' WARNING_CFLAGS ' , warn_array )
warn_array . add_item ( ' " -Wmost " ' )
warn_array . add_item ( ' " -Wno-four-char-constants " ' )
warn_array . add_item ( ' " -Wno-unknown-pragmas " ' )
bt_dict . add_item ( ' name ' , f ' " { buildtype } " ' )
# Now finally targets.
for target_name , target in self . build . get_build_targets ( ) . items ( ) :
self . generate_single_build_target ( objects_dict , target_name , target )
def generate_single_build_target ( self , objects_dict , target_name , target ) :
for buildtype in self . buildtypes :
dep_libs = [ ]
links_dylib = False
headerdirs = [ ]
for d in target . include_dirs :
for sd in d . incdirs :
cd = os . path . join ( d . curdir , sd )
headerdirs . append ( os . path . join ( self . environment . get_source_dir ( ) , cd ) )
headerdirs . append ( os . path . join ( self . environment . get_build_dir ( ) , cd ) )
for l in target . link_targets :
abs_path = os . path . join ( self . environment . get_build_dir ( ) ,
l . subdir , buildtype , l . get_filename ( ) )
dep_libs . append ( " ' %s ' " % abs_path )
if isinstance ( l , build . SharedLibrary ) :
links_dylib = True
if links_dylib :
dep_libs = [ ' -Wl,-search_paths_first ' , ' -Wl,-headerpad_max_install_names ' ] + dep_libs
dylib_version = None
if isinstance ( target , build . SharedLibrary ) :
ldargs = [ ' -dynamiclib ' , ' -Wl,-headerpad_max_install_names ' ] + dep_libs
install_path = os . path . join ( self . environment . get_build_dir ( ) , target . subdir , buildtype )
dylib_version = target . soversion
else :
ldargs = dep_libs
install_path = ' '
if dylib_version is not None :
product_name = target . get_basename ( ) + ' . ' + dylib_version
else :
product_name = target . get_basename ( )
ldargs + = target . link_args
linker , stdlib_args = self . determine_linker_and_stdlib_args ( target )
ldargs + = self . build . get_project_link_args ( linker , target . subproject , target . for_machine )
ldargs + = self . build . get_global_link_args ( linker , target . for_machine )
cargs = [ ]
for dep in target . get_external_deps ( ) :
cargs + = dep . get_compile_args ( )
ldargs + = dep . get_link_args ( )
ldstr = ' ' . join ( ldargs )
valid = self . buildconfmap [ target_name ] [ buildtype ]
langargs = { }
for lang in self . environment . coredata . compilers [ target . for_machine ] :
if lang not in LANGNAMEMAP :
continue
# Add compile args added using add_project_arguments()
pargs = self . build . projects_args [ target . for_machine ] . get ( target . subproject , { } ) . get ( lang , [ ] )
# Add compile args added using add_global_arguments()
# These override per-project arguments
gargs = self . build . global_args [ target . for_machine ] . get ( lang , [ ] )
targs = target . get_extra_args ( lang )
args = pargs + gargs + targs
if args :
langname = langnamemap [ lang ]
compiler = target . compilers . get ( lang )
lang_cargs = cargs
if compiler and target . implicit_include_directories :
lang_cargs + = self . get_build_dir_include_args ( target , compiler )
langargs [ langname ] = args
langargs [ langname ] + = lang_cargs
symroot = os . path . join ( self . environment . get_build_dir ( ) , target . subdir )
bt_dict = PbxDict ( )
objects_dict . add_item ( valid , bt_dict , buildtype )
bt_dict . add_item ( ' isa ' , ' XCBuildConfiguration ' )
settings_dict = PbxDict ( )
bt_dict . add_item ( ' buildSettings ' , settings_dict )
settings_dict . add_item ( ' COMBINE_HIDPI_IMAGES ' , ' YES ' )
if dylib_version is not None :
settings_dict . add_item ( ' DYLIB_CURRENT_VERSION ' , f ' " { dylib_version } ' )
settings_dict . add_item ( ' EXECUTABLE_PREFIX ' , f ' " { target . prefix } " ' )
if target . suffix == ' ' :
suffix = ' '
else :
suffix = ' . ' + target . suffix
settings_dict . add_item ( ' EXECUTABLE_SUFFIX ' , f ' " { suffix } " ' )
settings_dict . add_item ( ' GCC_GENERATE_DEBUGGING_SYMBOLS ' , BOOL2XCODEBOOL [ self . get_option_for_target ( OptionKey ( ' debug ' ) , target ) ] )
settings_dict . add_item ( ' GCC_INLINES_ARE_PRIVATE_EXTERN ' , ' NO ' )
settings_dict . add_item ( ' GCC_OPTIMIZATION_LEVEL ' , OPT2XCODEOPT [ self . get_option_for_target ( OptionKey ( ' optimization ' ) , target ) ] )
if target . has_pch :
# Xcode uses GCC_PREFIX_HEADER which only allows one file per target/executable. Precompiling various header files and
# applying a particular pch to each source file will require custom scripts (as a build phase) and build flags per each
# file. Since Xcode itself already discourages precompiled headers in favor of modules we don't try much harder here.
pchs = target . get_pch ( ' c ' ) + target . get_pch ( ' cpp ' ) + target . get_pch ( ' objc ' ) + target . get_pch ( ' objcpp ' )
# Make sure to use headers (other backends require implementation files like *.c *.cpp, etc; these should not be used here)
pchs = [ pch for pch in pchs if pch . endswith ( ' .h ' ) or pch . endswith ( ' .hh ' ) or pch . endswith ( ' hpp ' ) ]
if pchs :
if len ( pchs ) > 1 :
mlog . warning ( ' Unsupported Xcode configuration: More than 1 precompiled header found " {} " . Target " {} " might not compile correctly. ' . format ( str ( pchs ) , target . name ) )
relative_pch_path = os . path . join ( target . get_subdir ( ) , pchs [ 0 ] ) # Path relative to target so it can be used with "$(PROJECT_DIR)"
settings_dict . add_item ( ' GCC_PRECOMPILE_PREFIX_HEADER ' , ' YES ' )
settings_dict . add_item ( ' GCC_PREFIX_HEADER ' , f ' " $(PROJECT_DIR)/ { relative_pch_path } " ' )
settings_dict . add_item ( ' GCC_PREPROCESSOR_DEFINITIONS ' , ' " " ' )
settings_dict . add_item ( ' GCC_SYMBOLS_PRIVATE_EXTERN ' , ' NO ' )
if headerdirs :
quotedh = ' , ' . join ( [ ' " \\ " %s \\ " " ' % i for i in headerdirs ] )
settings_dict . add_item ( ' HEADER_SEARCH_PATHS ' , f ' ( { quotedh } ' )
settings_dict . add_item ( ' INSTALL_PATH ' , f ' " { install_path } " ' )
settings_dict . add_item ( ' LIBRARY_SEARCH_PATHS ' , ' " " ' )
if isinstance ( target , build . SharedLibrary ) :
settings_dict . add_item ( ' LIBRARY_STYLE ' , ' DYNAMIC ' )
for langname , args in langargs . items ( ) :
settings_dict . add_item ( f ' OTHER_ { langname } FLAGS ' , args )
settings_dict . add_item ( ' OTHER_LDFLAGS ' , f ' " { ldstr } " ' )
settings_dict . add_item ( ' OTHER_REZFLAGS ' , ' " " ' )
settings_dict . add_item ( ' PRODUCT_NAME ' , product_name )
settings_dict . add_item ( ' SECTORDER_FLAGS ' , ' " " ' )
settings_dict . add_item ( ' SYMROOT ' , f ' " { symroot } " ' )
settings_dict . add_item ( ' SYSTEM_HEADER_SEARCH_PATHS ' , ' " {} " ' . format ( self . environment . get_build_dir ( ) ) )
settings_dict . add_item ( ' USE_HEADERMAP ' , ' NO ' )
warn_array = PbxArray ( )
settings_dict . add_item ( ' WARNING_CFLAGS ' , warn_array )
warn_array . add_item ( ' " -Wmost " ' )
warn_array . add_item ( ' " -Wno-four-char-constants " ' )
warn_array . add_item ( ' " -Wno-unknown-pragmas " ' )
bt_dict . add_item ( ' name ' , buildtype )
def generate_xc_configurationList ( self , objects_dict ) :
# FIXME: sort items
conf_dict = PbxDict ( )
objects_dict . add_item ( self . project_conflist , conf_dict , f ' Build configuration list for PBXProject " { self . build . project_name } " ' )
conf_dict . add_item ( ' isa ' , ' XCConfigurationList ' )
confs_arr = PbxArray ( )
conf_dict . add_item ( ' buildConfigurations ' , confs_arr )
for buildtype in self . buildtypes :
confs_arr . add_item ( self . project_configurations [ self . buildtype ] , self . buildtype )
conf_dict . add_item ( ' defaultConfigurationIsVisible ' , 0 )
conf_dict . add_item ( ' defaultConfigurationName ' , self . buildtype )
# Now the all target
all_dict = PbxDict ( )
objects_dict . add_item ( self . all_buildconf_id , all_dict , ' Build configuration list for PBXAggregateTarget " ALL_BUILD " ' )
all_dict . add_item ( ' isa ' , ' XCConfigurationList ' )
conf_arr = PbxArray ( )
all_dict . add_item ( ' buildConfigurations ' , conf_arr )
for buildtype in self . buildtypes :
conf_arr . add_item ( self . buildall_configurations [ buildtype ] , buildtype )
all_dict . add_item ( ' defaultConfigurationIsVisible ' , 0 )
all_dict . add_item ( ' defaultConfigurationName ' , self . buildtype )
# Test target
test_dict = PbxDict ( )
objects_dict . add_item ( self . test_buildconf_id , test_dict , ' Build configuration list for PBXAggregateTarget " RUN_TEST " ' )
test_dict . add_item ( ' isa ' , ' XCConfigurationList ' )
conf_arr = PbxArray ( )
test_dict . add_item ( ' buildConfigurations ' , conf_arr )
for buildtype in self . buildtypes :
conf_arr . add_item ( self . test_configurations [ buildtype ] , buildtype )
test_dict . add_item ( ' defaultConfigurationIsVisible ' , 0 )
test_dict . add_item ( ' defaultConfigurationName ' , self . buildtype )
for target_name in self . build . get_build_targets ( ) :
t_dict = PbxDict ( )
listid = self . buildconflistmap [ target_name ]
objects_dict . add_item ( listid , t_dict , f ' Build configuration list for PBXNativeTarget " { target_name } " ' )
t_dict . add_item ( ' isa ' , ' XCConfigurationList ' )
conf_arr = PbxArray ( )
t_dict . add_item ( ' buildConfigurations ' , conf_arr )
idval = self . buildconfmap [ target_name ] [ self . buildtype ]
conf_arr . add_item ( idval , self . buildtype )
t_dict . add_item ( ' defaultConfigurationIsVisible ' , 0 )
t_dict . add_item ( ' defaultConfigurationName ' , self . buildtype )
def generate_prefix ( self , pbxdict ) :
pbxdict . add_item ( ' archiveVersion ' , ' 1 ' )
pbxdict . add_item ( ' classes ' , PbxDict ( ) )
pbxdict . add_item ( ' objectVersion ' , ' 46 ' )
objects_dict = PbxDict ( )
pbxdict . add_item ( ' objects ' , objects_dict )
return objects_dict
def generate_suffix ( self , pbxdict ) :
pbxdict . add_item ( ' rootObject ' , self . project_uid , ' Project object ' )