Blame cmake/MergeStaticLibs.cmake

Packit 1fb8d4
 
Packit 1fb8d4
#    Copyright (C) 2012 Modelon AB
Packit 1fb8d4
Packit 1fb8d4
#    This program is free software: you can redistribute it and/or modify
Packit 1fb8d4
#    it under the terms of the BSD style license.
Packit 1fb8d4
Packit 1fb8d4
#    This program is distributed in the hope that it will be useful,
Packit 1fb8d4
#    but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 1fb8d4
#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 1fb8d4
#    FMILIB_License.txt file for more details.
Packit 1fb8d4
Packit 1fb8d4
#    You should have received a copy of the FMILIB_License.txt file
Packit 1fb8d4
#    along with this program. If not, contact Modelon AB <http://www.modelon.com>.
Packit 1fb8d4
Packit 1fb8d4
# Merge_static_libs(output_library lib1 lib2 ... libn) merges a number of static
Packit 1fb8d4
# libs into a single static library
Packit 1fb8d4
function(merge_static_libs output_library)
Packit 1fb8d4
	set(output_target "${output_library}")
Packit 1fb8d4
	string(REGEX REPLACE "-" "_" output_library ${output_library})
Packit 1fb8d4
	set(libs ${ARGV})
Packit 1fb8d4
	list(REMOVE_AT libs 0)
Packit 1fb8d4
	
Packit 1fb8d4
	# Create a dummy file that the target will depend on
Packit 1fb8d4
	set(dummyfile ${CMAKE_CURRENT_BINARY_DIR}/${output_library}_dummy.c)
Packit 1fb8d4
	file(WRITE ${dummyfile} "const char * dummy = \"${dummyfile}\";")
Packit 1fb8d4
	
Packit 1fb8d4
	add_library(${output_target} STATIC ${dummyfile})
Packit 1fb8d4
Packit 1fb8d4
	if("${CMAKE_CFG_INTDIR}" STREQUAL ".")
Packit 1fb8d4
		set(multiconfig FALSE)
Packit 1fb8d4
	else()
Packit 1fb8d4
		set(multiconfig TRUE)
Packit 1fb8d4
	endif()
Packit 1fb8d4
	
Packit 1fb8d4
	# First get the file names of the libraries to be merged	
Packit 1fb8d4
	foreach(lib ${libs})
Packit 1fb8d4
		get_target_property(libtype ${lib} TYPE)
Packit 1fb8d4
		if(NOT libtype STREQUAL "STATIC_LIBRARY")
Packit 1fb8d4
			message(FATAL_ERROR "Merge_static_libs can only process static libraries")
Packit 1fb8d4
		endif()
Packit 1fb8d4
		if(multiconfig)
Packit 1fb8d4
			foreach(CONFIG_TYPE ${CMAKE_CONFIGURATION_TYPES})
Packit 1fb8d4
				get_target_property("libfile_${CONFIG_TYPE}" ${lib} "LOCATION_${CONFIG_TYPE}")
Packit 1fb8d4
				list(APPEND libfiles_${CONFIG_TYPE} ${libfile_${CONFIG_TYPE}})
Packit 1fb8d4
			endforeach()
Packit 1fb8d4
		else()
Packit 1fb8d4
			get_target_property(libfile ${lib} LOCATION)
Packit 1fb8d4
			list(APPEND libfiles "${libfile}")
Packit 1fb8d4
		endif(multiconfig)
Packit 1fb8d4
	endforeach()
Packit 1fb8d4
Packit 1fb8d4
	# Just to be sure: cleanup from duplicates
Packit 1fb8d4
	if(multiconfig)	
Packit 1fb8d4
		foreach(CONFIG_TYPE ${CMAKE_CONFIGURATION_TYPES})
Packit 1fb8d4
			list(REMOVE_DUPLICATES libfiles_${CONFIG_TYPE})
Packit 1fb8d4
			set(libfiles ${libfiles} ${libfiles_${CONFIG_TYPE}})
Packit 1fb8d4
		endforeach()
Packit 1fb8d4
	endif()
Packit 1fb8d4
	list(REMOVE_DUPLICATES libfiles)
Packit 1fb8d4
Packit 1fb8d4
	# Now the easy part for MSVC and for MAC
Packit 1fb8d4
  if(MSVC)
Packit 1fb8d4
    # lib.exe does the merging of libraries just need to conver the list into string
Packit 1fb8d4
	foreach(CONFIG_TYPE ${CMAKE_CONFIGURATION_TYPES})
Packit 1fb8d4
		set(flags "")
Packit 1fb8d4
		foreach(lib ${libfiles_${CONFIG_TYPE}})
Packit 1fb8d4
			set(flags "${flags} ${lib}")
Packit 1fb8d4
		endforeach()
Packit 1fb8d4
		string(TOUPPER "STATIC_LIBRARY_FLAGS_${CONFIG_TYPE}" PROPNAME)
Packit 1fb8d4
		set_target_properties(${output_target} PROPERTIES ${PROPNAME} "${flags}")
Packit 1fb8d4
	endforeach()
Packit 1fb8d4
	
Packit 1fb8d4
  elseif(APPLE)
Packit 1fb8d4
    # Use OSX's libtool to merge archives
Packit 1fb8d4
	if(multiconfig)
Packit 1fb8d4
		message(FATAL_ERROR "Multiple configurations are not supported")
Packit 1fb8d4
	endif()
Packit 1fb8d4
	get_target_property(outfile ${output_target} LOCATION)  
Packit 1fb8d4
	add_custom_command(TARGET ${output_target} POST_BUILD
Packit 1fb8d4
		COMMAND rm ${outfile}
Packit 1fb8d4
		COMMAND /usr/bin/libtool -static -o ${outfile} 
Packit 1fb8d4
		${libfiles}
Packit 1fb8d4
	)
Packit 1fb8d4
  else() 
Packit 1fb8d4
  # general UNIX - need to "ar -x" and then "ar -ru"
Packit 1fb8d4
	if(multiconfig)
Packit 1fb8d4
		message(FATAL_ERROR "Multiple configurations are not supported")
Packit 1fb8d4
	endif()
Packit 1fb8d4
	get_target_property(outfile ${output_target} LOCATION)
Packit 1fb8d4
	message(STATUS "output file location is ${outfile}")
Packit 1fb8d4
	foreach(lib ${libfiles})
Packit 1fb8d4
		# objlistfile will contain the list of object files for the library
Packit 1fb8d4
		set(objlistfile ${lib}.objlist)
Packit 1fb8d4
		set(objdir ${lib}.objdir)
Packit 1fb8d4
		set(objlistcmake  ${objlistfile}.cmake)
Packit 1fb8d4
		get_filename_component(libname ${lib} NAME_WE)
Packit 1fb8d4
Packit 1fb8d4
		if(${CMAKE_CURRENT_BINARY_DIR}/CMakeFiles/cmake.check_cache IS_NEWER_THAN ${objlistcmake})
Packit 1fb8d4
Packit 1fb8d4
			file(WRITE ${objlistcmake} "
Packit 1fb8d4
				# delete previous object files
Packit 1fb8d4
				message(STATUS \"Removing previous object files from ${lib}\")
Packit 1fb8d4
				EXECUTE_PROCESS(COMMAND ls .
Packit 1fb8d4
					WORKING_DIRECTORY ${objdir}
Packit 1fb8d4
					COMMAND xargs -I {} rm {}
Packit 1fb8d4
					WORKING_DIRECTORY ${objdir})
Packit 1fb8d4
				# Extract object files from the library
Packit 1fb8d4
				message(STATUS \"Extracting object files from ${lib}\")
Packit 1fb8d4
				EXECUTE_PROCESS(COMMAND ${CMAKE_AR} -x ${lib}
Packit 1fb8d4
					WORKING_DIRECTORY ${objdir})
Packit 1fb8d4
				# Prefixing object files to avoid conflicts
Packit 1fb8d4
				message(STATUS \"Prefixing object files to avoid conflicts\")
Packit 1fb8d4
				EXECUTE_PROCESS(COMMAND ls .
Packit 1fb8d4
					WORKING_DIRECTORY ${objdir}
Packit 1fb8d4
					COMMAND xargs -I {} mv {} ${libname}_{}
Packit 1fb8d4
					WORKING_DIRECTORY ${objdir})
Packit 1fb8d4
				# save the list of object files
Packit 1fb8d4
				EXECUTE_PROCESS(COMMAND ls . 
Packit 1fb8d4
					OUTPUT_FILE ${objlistfile}
Packit 1fb8d4
					WORKING_DIRECTORY ${objdir})
Packit 1fb8d4
			")
Packit 1fb8d4
				
Packit 1fb8d4
			file(MAKE_DIRECTORY ${objdir})
Packit 1fb8d4
				
Packit 1fb8d4
			add_custom_command(
Packit 1fb8d4
				OUTPUT ${objlistfile}
Packit 1fb8d4
				COMMAND ${CMAKE_COMMAND} -P ${objlistcmake}
Packit 1fb8d4
				DEPENDS ${lib})
Packit 1fb8d4
			
Packit 1fb8d4
		endif()
Packit 1fb8d4
		
Packit 1fb8d4
		list(APPEND extrafiles "${objlistfile}")
Packit 1fb8d4
		# relative path is needed by ar under MSYS
Packit 1fb8d4
		file(RELATIVE_PATH objlistfilerpath ${objdir} ${objlistfile})
Packit 1fb8d4
		add_custom_command(TARGET ${output_target} POST_BUILD
Packit 1fb8d4
			COMMAND ${CMAKE_COMMAND} -E echo "Running: ${CMAKE_AR} ru ${outfile} @${objlistfilerpath}"
Packit 1fb8d4
			COMMAND ${CMAKE_AR} ru "${outfile}" @"${objlistfilerpath}"
Packit 1fb8d4
			#COMMAND ld -r -static -o "${outfile}" --whole-archive @"${objlistfilerpath}"
Packit 1fb8d4
			WORKING_DIRECTORY ${objdir})
Packit 1fb8d4
	endforeach()
Packit 1fb8d4
	add_custom_command(TARGET ${output_target} POST_BUILD
Packit 1fb8d4
			COMMAND ${CMAKE_COMMAND} -E echo "Running: ${CMAKE_RANLIB} ${outfile}"
Packit 1fb8d4
			COMMAND ${CMAKE_RANLIB} ${outfile})
Packit 1fb8d4
  endif()
Packit 1fb8d4
  file(WRITE ${dummyfile}.base "const char* ${output_library}_sublibs=\"${libs}\";")
Packit 1fb8d4
  add_custom_command( 
Packit 1fb8d4
		OUTPUT ${dummyfile}
Packit 1fb8d4
		COMMAND ${CMAKE_COMMAND} -E copy ${dummyfile}.base ${dummyfile}
Packit 1fb8d4
		DEPENDS ${libs} ${extrafiles})
Packit 1fb8d4
Packit 1fb8d4
endfunction()