blob: 2f7639f489cb01369a23e53ebd0580d32ae639d1 [file] [log] [blame]
Mate Toth-Pal76867262018-03-09 13:15:36 +01001#-------------------------------------------------------------------------------
Sverteczky, Marcell4b78a4b2019-06-03 14:17:10 +02002# Copyright (c) 2017-2019, Arm Limited. All rights reserved.
Mate Toth-Pal76867262018-03-09 13:15:36 +01003#
4# SPDX-License-Identifier: BSD-3-Clause
5#
6#-------------------------------------------------------------------------------
7
8#This file contains settings to specify how GNUARM shall be used
9
10function(check_gnuarm_input_vars MY_VERSION)
11 #Specify where gnuarm is
12 if (NOT DEFINED GNUARM_PATH)
13 message(FATAL_ERROR "Please set GNUARM_PATH to the root directory of the gnuarm installation. e.g. set(GNUARM_PATH \"C:/Program Files (x86)/GNU Tools ARM Embedded/6 2017-q1-update/\"")
14 endif()
15
16 STRING(REGEX REPLACE "([0-9]+).([0-9]+).*" "\\1.\\2" _MY_MAJOR_MINOR "${MY_VERSION}")
17 STRING(REGEX REPLACE "([0-9]+).([0-9]+).*" "\\1.\\2" _GNUARM_MAJOR_MINOR "${GNUARM_VER}")
18
19 #Check gnuarm version.
20 if (NOT "${_MY_MAJOR_MINOR}" VERSION_EQUAL "${_GNUARM_MAJOR_MINOR}")
21 message(FATAL_ERROR "GNUARM version (GNUARM_VER=${GNUARM_VER}) does not match ${MY_VERSION}")
22 endif()
23
Antonio de Angelisf1f7ebd2018-11-23 23:11:41 +000024 if (NOT DEFINED ARM_CPU_ARCHITECTURE AND NOT DEFINED ARM_CPU_TYPE)
25 message(FATAL_ERROR "ARM_CPU_TYPE and ARM_CPU_ARCHITECTURE is not defined! Please include the CPU specific config file before this one.")
Mate Toth-Pal76867262018-03-09 13:15:36 +010026 endif()
27
28endfunction()
29
30message(STATUS "Using gcc compiler package v${GNUARM_VER} from ${GNUARM_PATH}")
31
32
33#Tell cmake which compiler we use
34if (EXISTS "c:/")
35 set (CMAKE_C_COMPILER "${GNUARM_PATH}/bin/arm-none-eabi-gcc.exe")
36 set (CMAKE_CXX_COMPILER "${GNUARM_PATH}/bin/arm-none-eabi-g++.exe")
37 set (CMAKE_ASM_COMPILER "${GNUARM_PATH}/bin/arm-none-eabi-gcc.exe")
38else()
39 set (CMAKE_C_COMPILER "${GNUARM_PATH}/bin/arm-none-eabi-gcc")
40 set (CMAKE_CXX_COMPILER "${GNUARM_PATH}/bin/arm-none-eabi-g++")
41 set (CMAKE_ASM_COMPILER "${GNUARM_PATH}/bin/arm-none-eabi-gcc")
42endif()
43
44if("CXX" IN_LIST languages)
45 set(CMAKE_CXX_COMPILER_ID "GNUARM" CACHE INTERNAL "CXX compiler ID" FORCE)
46 include(Compiler/GNUARM-CXX)
47endif()
48
49if("C" IN_LIST languages)
50 set(CMAKE_C_COMPILER_ID "GNUARM" CACHE INTERNAL "C compiler ID" FORCE)
51 include(Compiler/GNUARM-C)
52endif()
53
54if("ASM" IN_LIST languages)
55 set(CMAKE_C_COMPILER_ID "GNUARM" CACHE INTERNAL "ASM compiler ID" FORCE)
56 include(Compiler/GNUARM-ASM)
57endif()
58
59function(compiler_set_linkercmdfile)
60 set( _OPTIONS_ARGS ) #Option (on/off) arguments.
61 set( _ONE_VALUE_ARGS TARGET PATH) #Single option arguments.
62 set( _MULTI_VALUE_ARGS DEFINES INCLUDES) #List arguments
63 cmake_parse_arguments(_MY_PARAMS "${_OPTIONS_ARGS}" "${_ONE_VALUE_ARGS}" "${_MULTI_VALUE_ARGS}" ${ARGN} )
64
65 #Check passed parameters
66 if(NOT _MY_PARAMS_TARGET)
67 message(FATAL_ERROR "compiler_set_linkercmdfile: mandatory parameter 'TARGET' is missing.")
68 endif()
69 if (NOT TARGET ${_MY_PARAMS_TARGET})
70 message(FATAL_ERROR "compiler_set_linkercmdfile: value of parameter 'TARGET' is invalid.")
71 endif()
72
73 if(NOT _MY_PARAMS_PATH)
74 message(FATAL_ERROR "compiler_set_linkercmdfile: mandatory parameter 'PATH' is missing.")
75 endif()
76 set(_FILE_PATH ${_MY_PARAMS_PATH})
77
Mate Toth-Pal76867262018-03-09 13:15:36 +010078 #Create additional target if linker script needs to be pre-processed.
79 if (_MY_PARAMS_DEFINES OR _MY_PARAMS_INCLUDES)
80 #Name of pre-processed linker script file.
Tamas Ban6032c3b2018-07-17 22:11:20 +010081 set(FINAL_LD_FILE_NAME "${CMAKE_CURRENT_BINARY_DIR}/${_MY_PARAMS_TARGET}.ld.i")
Mate Toth-Pal76867262018-03-09 13:15:36 +010082 #Name of the target doing the pre-processing
83 set(LD_PP_TARGET_NAME "${_MY_PARAMS_TARGET}_ldpp")
Sverteczky, Marcell4b78a4b2019-06-03 14:17:10 +020084 compiler_preprocess_file(SRC ${_MY_PARAMS_PATH}
85 DST ${FINAL_LD_FILE_NAME}
86 TARGET_PREFIX ${_MY_PARAMS_TARGET}
87 BEFORE_TARGET ${_MY_PARAMS_TARGET}
88 DEFINES ${_MY_PARAMS_DEFINES}
89 INCLUDES ${_MY_PARAMS_INCLUDES})
90
91 #Tell cmake to delete the intermediate linker script when the clean rule is executed.
Mate Toth-Pal76867262018-03-09 13:15:36 +010092 get_directory_property(_ADDITIONAL_MAKE_CLEAN_FILES DIRECTORY "./" ADDITIONAL_MAKE_CLEAN_FILES)
93 set_directory_properties(PROPERTY ADDITIONAL_MAKE_CLEAN_FILES "${_ADDITIONAL_MAKE_CLEAN_FILES} ${FINAL_LD_FILE_NAME}")
94 #Set the path to linker script point to the intermediate file.
95 set(_FILE_PATH ${FINAL_LD_FILE_NAME})
96 endif()
97
98 #Note: the space before the option is important!
99 set_property(TARGET ${_MY_PARAMS_TARGET} APPEND_STRING PROPERTY LINK_FLAGS " -T${_FILE_PATH}")
100 set_property(TARGET ${_MY_PARAMS_TARGET} APPEND PROPERTY LINK_DEPENDS ${_FILE_PATH})
101 #Tell cmake .map files shall be removed when project is cleaned (make clean)
102 get_filename_component(_TARGET_BASE_NAME ${_MY_PARAMS_TARGET} NAME_WE)
103 get_directory_property(_ADDITIONAL_MAKE_CLEAN_FILES DIRECTORY "./" ADDITIONAL_MAKE_CLEAN_FILES)
104 set_directory_properties(PROPERTY ADDITIONAL_MAKE_CLEAN_FILES "${_ADDITIONAL_MAKE_CLEAN_FILES} ${_TARGET_BASE_NAME}.map")
105endfunction()
106
107function(compiler_set_cmse_output TARGET FILE_PATH)
108 #Note: the space before the option is important!
109 set_property(TARGET ${TARGET} APPEND_STRING PROPERTY LINK_FLAGS " -Wl,--cmse-implib,--out-implib=${FILE_PATH}")
110 #Tell cmake cmse output is a generated object file.
111 SET_SOURCE_FILES_PROPERTIES("${FILE_PATH}" PROPERTIES EXTERNAL_OBJECT true GENERATED true)
112 #Tell cmake cmse output shall be removed by clean target.
113 get_directory_property(_ADDITIONAL_MAKE_CLEAN_FILES DIRECTORY "./" ADDITIONAL_MAKE_CLEAN_FILES)
114 set_directory_properties(PROPERTY ADDITIONAL_MAKE_CLEAN_FILES "${_ADDITIONAL_MAKE_CLEAN_FILES} ${FILE_PATH}")
115endfunction()
116
117function(compiler_merge_library)
118 set( _OPTIONS_ARGS ) #Option (on/off) arguments.
119 set( _ONE_VALUE_ARGS DEST) #Single option arguments.
120 set( _MULTI_VALUE_ARGS LIBS) #List arguments
121 cmake_parse_arguments(_MY_PARAMS "${_OPTIONS_ARGS}" "${_ONE_VALUE_ARGS}" "${_MULTI_VALUE_ARGS}" ${ARGN} )
122
123 #Check passed parameters
124 if(NOT _MY_PARAMS_DEST)
125 message(FATAL_ERROR "compiler_merge_library: no destination target specified. Please see the DEST parameter.")
126 endif()
127 #Check if destination is a library
128 get_target_property(_tmp ${_MY_PARAMS_DEST} TYPE)
129 if(NOT "${_tmp}" STREQUAL "STATIC_LIBRARY")
130 message(FATAL_ERROR "compiler_merge_library: parameter DEST must be a static library target.")
131 endif()
132 set(_DEST ${_MY_PARAMS_DEST})
133
134 if(NOT _MY_PARAMS_LIBS)
135 message(FATAL_ERROR "compiler_merge_library: no source libraries specified. Please see the LIBS parameter.")
136 endif()
137 set(_LIBS ${_MY_PARAMS_LIBS})
138
139 ##Find the cmake script doing the merge.
140 find_file(_MERGE_SCRIPT "GNUArMerge.cmake" PATHS ${CMAKE_MODULE_PATH} PATH_SUFFIXES Common NO_DEFAULT_PATH)
141
142 #Now add a custom command for each source library to our custom target to
143 #merge into the destination.
144 foreach(SRC_LIB ${_LIBS})
145 get_filename_component(_SRC_LIB_NAME "${SRC_LIB}" NAME)
146 add_custom_command(TARGET ${_DEST} POST_BUILD
147 COMMAND ${CMAKE_COMMAND} -DCMAKE_AR=${CMAKE_AR} -DSOURCE=${SRC_LIB} -DDESTINATION=$<TARGET_FILE:${_DEST}> -P ${_MERGE_SCRIPT}
148 COMMENT "\t\tmerging objects from ${_SRC_LIB_NAME}")
149 endforeach()
150endfunction()
151
152function(compiler_generate_binary_output TARGET)
153 add_custom_command(TARGET ${TARGET} POST_BUILD COMMAND ${CMAKE_GNUARM_OBJCOPY} ARGS -O binary $<TARGET_FILE:${TARGET}> $<TARGET_FILE_DIR:${TARGET}>/${TARGET}.bin)
154endfunction()
Sverteczky, Marcell4b78a4b2019-06-03 14:17:10 +0200155
156# Function for creating a new target that preprocesses a .c file
157#INPUTS:
158# SRC - (mandatory) - file to be preprocessed
159# DST - (mandatory) - output file for the preprocessing
160# TARGET_PREFIX - (optional) - prefix for the target that this function creates and which manages the preprocessing
161# BEFORE_TARGET - (optional) - target which is dependent on the preprocessing target in the below function
162# DEFINES - (optional) - additional command line switches from macro definitions for preprocessing
163# INCLUDES - (optional) - additional command line switches from include paths for preprocessing
164function(compiler_preprocess_file)
165 #Option (on/off) arguments.
166 set( _OPTIONS_ARGS)
167 #Single option arguments.
168 set( _ONE_VALUE_ARGS SRC DST TARGET_PREFIX BEFORE_TARGET)
169 #List arguments
170 set( _MULTI_VALUE_ARGS DEFINES INCLUDES)
171 cmake_parse_arguments(_MY_PARAMS "${_OPTIONS_ARGS}" "${_ONE_VALUE_ARGS}" "${_MULTI_VALUE_ARGS}" ${ARGN} )
172
173 #Check passed parameters
174 if(NOT DEFINED _MY_PARAMS_SRC)
175 message(FATAL_ERROR "compiler_preprocess_file: mandatory parameter 'SRC' is missing.")
176 endif()
177
178 if(NOT DEFINED _MY_PARAMS_DST)
179 message(FATAL_ERROR "compiler_preprocess_file: mandatory parameter 'DST' is missing.")
180 endif()
181
182 if(DEFINED _MY_PARAMS_BEFORE_TARGET)
183 if(NOT TARGET ${_MY_PARAMS_BEFORE_TARGET})
184 message(FATAL_ERROR "compiler_preprocess_file: optional parameter 'BEFORE_TARGET' is not target.")
185 endif()
186 endif()
187
188 #Compose additional command line switches from macro definitions.
189 set(_FLAGS "")
190 if (_MY_PARAMS_DEFINES)
191 foreach(_DEFINE IN LISTS _MY_PARAMS_DEFINES)
192 list(APPEND _FLAGS "-D${_DEFINE}")
193 endforeach()
194 endif()
195
196 #Compose additional command line switches from include paths.
197 if (_MY_PARAMS_INCLUDES)
198 foreach(_INCLUDE IN LISTS _MY_PARAMS_INCLUDES)
199 list(APPEND _FLAGS "-I${_INCLUDE}")
200 endforeach()
201 endif()
202
203 #The compiler flag might contain leading spaces which can fail the preprocess operation, these are removed
204 STRING(STRIP ${CMAKE_C_FLAGS_CPU} _MY_TEMP_CMAKE_C_FLAGS_CPU)
205 #If a string contains spaces, then it is inserted amongst quotation marks. Furthermore the compiler fails if it is
206 #called with multiple switches included in one quotation mark. If the extra spaces are replaced by semicolons,
207 #then the insertion will be advantageous for the compiler.
208 STRING(REPLACE " " ";" _MY_TEMP2_CMAKE_C_FLAGS_CPU ${_MY_TEMP_CMAKE_C_FLAGS_CPU})
209 set(_LOCAL_CMAKE_C_FLAGS_CPU "")
210 foreach(_C_FLAG IN LISTS _MY_TEMP2_CMAKE_C_FLAGS_CPU)
211 list(APPEND _LOCAL_CMAKE_C_FLAGS_CPU "${_C_FLAG}")
212 endforeach()
213
214 add_custom_command(OUTPUT ${_MY_PARAMS_DST}
215 COMMAND ${CMAKE_C_COMPILER} ${_LOCAL_CMAKE_C_FLAGS_CPU} -E -P -xc ${_FLAGS} ${_MY_PARAMS_SRC} -o ${_MY_PARAMS_DST}
216 DEPENDS ${_MY_PARAMS_SRC}
217 COMMENT "Preprocess the ${_MY_PARAMS_SRC} file"
218 )
219
220 set(_MY_TARGET_PREFIX "")
221 if(TARGET ${_MY_PARAMS_TARGET_PREFIX})
222 set(_MY_TARGET_PREFIX "${_MY_PARAMS_TARGET_PREFIX}")
223 endif()
224 #The preprocessing related target name is obtained by indexing the file's name that is to be preprocessed
225 get_filename_component(_MY_FILENAME_TO_BE_INDEXED ${_MY_PARAMS_SRC} NAME_WE)
226 foreach(_SUFFIX RANGE 1 100)
227 if (NOT TARGET ${_MY_TARGET_PREFIX}_pp_${_MY_FILENAME_TO_BE_INDEXED}_${_SUFFIX})
228 set(_PREPROCESS_TARGET_NAME "${_MY_TARGET_PREFIX}_pp_${_MY_FILENAME_TO_BE_INDEXED}_${_SUFFIX}")
229 break()
230 endif()
231 if (_SUFFIX EQUAL 100)
232 message(FATAL_ERROR "You have called 'compiler_preprocess_file' too many times (${_SUFFIX} function calls).")
233 endif()
234 endforeach()
235
236 #Make the original target depend on the new one.
237 if(TARGET ${_MY_PARAMS_BEFORE_TARGET})
238 add_custom_target(${_PREPROCESS_TARGET_NAME} DEPENDS ${_MY_PARAMS_DST})
239 add_dependencies(${_MY_PARAMS_BEFORE_TARGET} ${_PREPROCESS_TARGET_NAME})
240 else()
241 add_custom_target(${_PREPROCESS_TARGET_NAME} ALL DEPENDS ${_MY_PARAMS_DST})
242 endif()
243endfunction()