# Copyright 2001 David Abrahams. # Copyright 2002-2006 Rene Rivera. # Copyright 2002-2003 Vladimir Prus. # # Distributed under the Boost Software License, Version 1.0. # (See accompanying file LICENSE_1_0.txt or copy at # http://www.boost.org/LICENSE_1_0.txt) import toolset : flags ; import property ; import generators ; import os ; import type ; import feature ; import "class" : new ; import set ; import common ; import errors ; if [ MATCH (--debug-configuration) : [ modules.peek : ARGV ] ] { .debug-configuration = true ; } feature.extend toolset : gcc ; import unix ; toolset.inherit-generators gcc : unix : unix.link unix.link.dll ; toolset.inherit-flags gcc : unix ; toolset.inherit-rules gcc : unix ; generators.override gcc.prebuilt : builtin.prebuilt ; generators.override gcc.searched-lib-generator : searched-lib-generator ; # Make the "o" suffix used for gcc toolset on all # platforms type.set-generated-target-suffix OBJ : gcc : o ; type.set-generated-target-suffix STATIC_LIB : gcc : a ; import rc ; # Initializes the gcc toolset for the given version. # If necessary, command may be used to specify where the compiler # is located. # The parameter 'options' is a space-delimited list of options, each # one being specified as option-value. Valid option names # are: cxxflags, linkflags and linker-type. Accepted values for linker-type # are gnu and sun, gnu being the default. # Example: # using gcc : 3.4 : : foo bar sun ; rule init ( version ? : command * : options * ) { local condition = [ common.check-init-parameters gcc : version $(version) ] ; local command = [ common.get-invocation-command gcc : g++ : $(command) ] ; common.handle-options gcc : $(condition) : $(command) : $(options) ; local linker = [ feature.get-values : $(options) ] ; if ! $(linker) { if [ os.name ] = OSF { linker = osf ; } else { linker = gnu ; } } init-link-flags gcc $(linker) $(condition) ; local root = [ feature.get-values : $(options) ] ; local bin ; if $(command) { bin ?= [ common.get-absolute-tool-path $(command[-1]) ] ; root ?= $(bin:D) ; } # If gcc is installed in non-standard location, we'd need to # add LD_LIBRARY_PATH when running programs created with it # (for unit-test/run rules). if $(command) { # On multilib 64-bit boxes, there are both 32-bit and 64-bit # libraries and all must be added to LD_LIBRARY_PATH. The linker # will pick the right onces. # Note that we don't provide a clean way to build 32-bit binary # with 64-bit compiler, but user can always pass -m32 manually. local lib_path = $(root)/bin $(root)/lib $(root)/lib32 $(root)/lib64 ; if $(.debug-configuration) { ECHO notice: using gcc libraries :: $(condition) :: $(lib_path) ; } flags gcc.link RUN_PATH $(condition) : $(lib_path) ; } #~ If it's not a system gcc install we should adjust the various #~ programs as needed to prefer using the install specific versions. #~ This is essential for correct use of MinGW and for cross-compiling. #~ - The archive builder. local archiver = [ common.get-invocation-command gcc : ar : [ feature.get-values : $(options) ] : $(bin) : search-path ] ; flags gcc.archive .AR $(condition) : $(archiver[1]) ; if $(.debug-configuration) { ECHO notice: using gcc archiver :: $(condition) :: $(archiver[1]) ; } #~ - The resource compiler. local rc = [ common.get-invocation-command gcc : windres : [ feature.get-values : $(options) ] : $(bin) : search-path ] ; local rc-type = [ feature.get-values : $(options) ] ; rc-type ?= windres ; if ! $(rc) { #~ If we can't find an RC compiler we fallback to a null RC compiler #~ that creates empty object files. This allows the same Jamfiles #~ to work across the board. The null RC uses the assembler to create #~ the empty objects, so configure that. rc = [ common.get-invocation-command gcc : as : : $(bin) : search-path ] ; rc-type = null ; } rc.configure $(rc) : $(condition) : $(rc-type) ; } if [ os.name ] = NT { # This causes single-line command invocation to not go through # .bat files, thus avoiding command-line length limitations JAMSHELL = % ; } generators.register-c-compiler gcc.compile.c++ : CPP : OBJ : gcc ; generators.register-c-compiler gcc.compile.c : C : OBJ : gcc ; generators.register-c-compiler gcc.compile.asm : ASM : OBJ : gcc ; # Declare flags and action for compilation flags gcc.compile OPTIONS off : -O0 ; flags gcc.compile OPTIONS speed : -O3 ; flags gcc.compile OPTIONS space : -Os ; flags gcc.compile OPTIONS off : -fno-inline ; flags gcc.compile OPTIONS on : -Wno-inline ; flags gcc.compile OPTIONS full : -finline-functions -Wno-inline ; flags gcc.compile OPTIONS off : -w ; flags gcc.compile OPTIONS on : -Wall ; flags gcc.compile OPTIONS all : -Wall -pedantic ; flags gcc.compile OPTIONS on : -Werror ; flags gcc.compile OPTIONS on : -g ; flags gcc.compile OPTIONS on : -pg ; # On cygwin and mingw, gcc generates position independent code by default, # and warns if -fPIC is specified. This might not be the right way # of checking if we're using cygwin. For example, it's possible # to run cygwin gcc from NT shell, or using crosscompiling. # But we'll solve that problem when it's time. In that case # we'll just add another parameter to 'init' and move this login # inside 'init'. if [ os.name ] != CYGWIN && [ os.name ] != NT { # This logic will add -fPIC for all compilations: # # lib a : a.cpp b ; # obj b : b.cpp ; # exe c : c.cpp a d ; # obj d : d.cpp ; # # This all is fine, except that 'd' will be compiled with # -fPIC even though it's not needed, as 'd' is used only in # exe. However, it's hard to detect where a target is going to # be used. Alternative, we can set -fPIC only when main target type # is LIB but than 'b' will be compiled without -fPIC. In x86-64 that # will lead to link errors. So, compile everything with -fPIC. # # Yet another alternative would be to create propagated # feature, and set it when building shared libraries, but that's hard # to implement and will increase target path length even more. flags gcc.compile OPTIONS shared : -fPIC ; } if [ os.name ] != NT && [ os.name ] != OSF { # OSF does have an option called -soname but it doesn't seem to work as # expected, therefore it has been disabled. HAVE_SONAME = "" ; SONAME_OPTION = -h ; } flags gcc.compile USER_OPTIONS ; flags gcc.compile.c++ USER_OPTIONS ; flags gcc.compile DEFINES ; flags gcc.compile INCLUDES ; rule compile.c++ { # Some extensions are compiled as C++ by default. For others, we need # to pass -x c++. # We could always pass -x c++ but distcc does not work with it. if ! $(>:S) in .cc .cp .cxx .cpp .c++ .C { LANG on $(<) = "-x c++" ; } } actions compile.c++ { "$(CONFIG_COMMAND)" $(LANG) -ftemplate-depth-128 $(OPTIONS) $(USER_OPTIONS) -D$(DEFINES) -I"$(INCLUDES)" -c -o "$(<)" "$(>)" } rule compile.c { # If we use the name g++ then default file suffix -> language mapping # does not work. So have to pass -x option. Maybe, we can work around this # by allowing the user to specify both C and C++ compiler names. #if $(>:S) != .c #{ LANG on $(<) = "-x c" ; #} } actions compile.c { "$(CONFIG_COMMAND)" $(LANG) $(OPTIONS) $(USER_OPTIONS) -D$(DEFINES) -I"$(INCLUDES)" -c -o "$(<)" "$(>)" } rule compile.asm { LANG on $(<) = "-x assembler-with-cpp" ; } actions compile.asm { "$(CONFIG_COMMAND)" $(LANG) $(OPTIONS) -D$(DEFINES) -I"$(INCLUDES)" -c -o "$(<)" "$(>)" } # The class which check that we don't try to use # the static property while creating or using shared library, # since it's not supported by gcc/libc. class gcc-linking-generator : unix-linking-generator { rule run ( project name ? : property-set : sources + ) { #~ TODO: Replace this with the use of a target-os property. local no-static-link = ; if [ modules.peek : UNIX ] { switch [ modules.peek : JAMUNAME ] { case * : no-static-link = true ; } } local properties = [ $(property-set).raw ] ; local reason ; if $(no-static-link) && static in $(properties) { if shared in $(properties) { reason = "On gcc, DLL can't be build with 'static'." ; } else if [ type.is-derived $(self.target-types[1]) EXE ] { for local s in $(sources) { local type = [ $(s).type ] ; if $(type) && [ type.is-derived $(type) SHARED_LIB ] { reason = "On gcc, using DLLS together with the" "static options is not possible " ; } } } } if $(reason) { ECHO warning: $(reason) ; ECHO warning: "It's suggested to use 'static' together" "with the 'static'." ; return ; } else { return [ unix-linking-generator.run $(project) $(name) : $(property-set) : $(sources) ] ; } } } generators.register [ new gcc-linking-generator gcc.link : LIB OBJ : EXE : gcc ] ; generators.register [ new gcc-linking-generator gcc.link.dll : LIB OBJ : SHARED_LIB : gcc ] ; # Declare flags for linking # First, the common flags flags gcc.link OPTIONS on : -g ; flags gcc.link OPTIONS on : -pg ; flags gcc.link USER_OPTIONS ; flags gcc.link LINKPATH ; flags gcc.link FINDLIBS-ST ; flags gcc.link FINDLIBS-SA ; flags gcc.link LIBRARIES ; # For static we made sure there are no dynamic libraries # in the link flags gcc.link OPTIONS static : -static ; # Now, the vendor specific flags # The parameter linker can be either gnu or sun rule init-link-flags ( toolset linker condition ) { switch $(linker) { case gnu : { # Strip the binary when no debugging is needed. # We use --strip-all flag as opposed to -s since icc # (intel's compiler) is generally option-compatible with # and inherits from gcc toolset, but does not support -s flags $(toolset).link OPTIONS $(condition)/off : -Wl,--strip-all : unchecked ; flags $(toolset).link RPATH $(condition) : : unchecked ; flags $(toolset).link RPATH_LINK $(condition) : : unchecked ; flags $(toolset).link START-GROUP $(condition) : -Wl,--start-group : unchecked ; flags $(toolset).link END-GROUP $(condition) : -Wl,--end-group : unchecked ; } case darwin : { # On Darwin, the -s option to ld does not work unless we pass # -static, and passing -static unconditionally is a bad idea. # So, don't pass -s at all, darwin.jam will use separate 'strip' # invocation. flags $(toolset).link RPATH $(condition) : : unchecked ; flags $(toolset).link RPATH_LINK $(condition) : : unchecked ; } case osf : { # No --strip-all, just -s flags $(toolset).link OPTIONS $(condition)/off : -Wl,-s : unchecked ; flags $(toolset).link RPATH $(condition) : : unchecked ; # This does not supports -R flags $(toolset).link RPATH_OPTION $(condition) : -rpath : unchecked ; # -rpath-link is not supported at all. } case sun : { flags $(toolset).link OPTIONS $(condition)/off : -Wl,-s : unchecked ; flags $(toolset).link RPATH $(condition) : : unchecked ; # Solaris linker does not have a separate -rpath-link, but # allows to use -L for the same purpose. flags $(toolset).link LINKPATH $(condition) : : unchecked ; # This permits shared libraries with non-PIC code on Solaris # VP, 2004/09/07: Now that we have -fPIC hardcode in link.dll, # the following is not needed. Whether -fPIC should be hardcoded, # is a separate question. # AH, 2004/10/16: it is still necessary because some tests link # against static libraries that were compiled without PIC. flags $(toolset).link OPTIONS $(condition)/shared : -mimpure-text : unchecked ; } case * : { errors.user-error "$(toolset) initialization: invalid linker '$(linker)'" : "The value '$(linker)' specified for is not recognized." : "Possible values are 'sun', 'gnu'" ; } } } # Declare actions for linking rule link ( targets * : sources * : properties * ) { SPACE on $(targets) = " " ; # Serialize execution of the 'link' action, since # running N links in parallel is just slower. # For now, serialize only gcc links, it might be a good # idea to serialize all links. JAM_SEMAPHORE on $(targets) = gcc-link-semaphore ; } actions link bind LIBRARIES { "$(CONFIG_COMMAND)" -L"$(LINKPATH)" -Wl,$(RPATH_OPTION:E=-R)$(SPACE)-Wl,"$(RPATH)" -Wl,-rpath-link$(SPACE)-Wl,"$(RPATH_LINK)" -o "$(<)" $(START-GROUP) "$(>)" "$(LIBRARIES)" -l$(FINDLIBS-ST) -l$(FINDLIBS-SA) $(END-GROUP) $(OPTIONS) $(USER_OPTIONS) } # Default value. Mostly for the sake of intel-linux # that inherits from gcc, but does not has the same # logic to set the .AR variable. We can put the same # logic in intel-linux, but that's hardly worth the trouble # as on Linux, 'ar' is always available. .AR = ar ; flags gcc.archive AROPTIONS ; rule archive ( targets * : sources * : properties * ) { # Always remove archive and start again. Here's rationale from # # Andre Hentz: # # I had a file, say a1.c, that was included into liba.a. # I moved a1.c to a2.c, updated my Jamfiles and rebuilt. # My program was crashing with absurd errors. # After some debugging I traced it back to the fact that a1.o was *still* # in liba.a # # Rene Rivera: # # Originally removing the archive was done by splicing an RM # onto the archive action. That makes archives fail to build on NT # when they have many files because it will no longer execute the # action directly and blow the line length limit. Instead we # remove the file in a different action, just before the building # of the archive. # local clean.a = $(targets[1])(clean) ; TEMPORARY $(clean.a) ; NOCARE $(clean.a) ; LOCATE on $(clean.a) = [ on $(targets[1]) return $(LOCATE) ] ; DEPENDS $(clean.a) : $(sources) ; DEPENDS $(targets) : $(clean.a) ; common.RmTemps $(clean.a) : $(targets) ; } # Declare action for creating static libraries # The 'r' letter means to add files to the archive with replacement # Since we remove archive, we don't care about replacement, but # there's no option "add without replacement". # The 'c' letter means suppresses warning in case the archive # does not exists yet. That warning is produced only on # some platforms, for whatever reasons. actions piecemeal archive { "$(.AR)" $(AROPTIONS) rc "$(<)" "$(>)" } rule link.dll ( targets * : sources * : properties * ) { SPACE on $(targets) = " " ; JAM_SEMAPHORE on $(targets) = gcc-link-semaphore ; } # Differ from 'link' above only by -shared. actions link.dll bind LIBRARIES { "$(CONFIG_COMMAND)" -L"$(LINKPATH)" -Wl,$(RPATH_OPTION:E=-R)$(SPACE)-Wl,"$(RPATH)" -o "$(<)" $(HAVE_SONAME)-Wl,$(SONAME_OPTION)$(SPACE)-Wl,$(<[1]:D=) -shared $(START-GROUP) "$(>)" "$(LIBRARIES)" -l$(FINDLIBS-ST) -l$(FINDLIBS-SA) $(END-GROUP) $(OPTIONS) $(USER_OPTIONS) } # Set up threading support. It's somewhat contrived, so perform it at the end, # to avoid cluttering other code. if [ os.on-windows ] { flags gcc OPTIONS multi : -mthreads ; } else if [ modules.peek : UNIX ] { switch [ modules.peek : JAMUNAME ] { case SunOS* : { flags gcc OPTIONS multi : -pthreads ; flags gcc FINDLIBS-SA multi : rt ; } case BeOS : { # BeOS has no threading options, don't set anything here. } case *BSD : { flags gcc OPTIONS multi : -pthread ; # there is no -lrt on BSD } case DragonFly : { flags gcc OPTIONS multi : -pthread ; # there is no -lrt on BSD - DragonFly is a FreeBSD variant, # which anoyingly doesn't say it's a *BSD. } case IRIX : { # gcc on IRIX does not support multi-threading, don't set anything here. } case HP_UX : { # gcc on HP-UX does not support multi-threading, don't set anything here } case Darwin : { # Darwin has no threading options, don't set anything here. } case * : { flags gcc OPTIONS multi : -pthread ; flags gcc FINDLIBS-SA multi : rt ; } } }