-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathBloaty.cmake
226 lines (197 loc) · 8.06 KB
/
Bloaty.cmake
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
#
# Copyright (C) 2021 Swift Navigation Inc.
# Contact: Swift Navigation <[email protected]>
#
# This source is subject to the license found in the file 'LICENSE' which must
# be be distributed together with this source. All other rights reserved.
#
# THIS CODE AND INFORMATION IS PROVIDED "AS IS" WITHOUT WARRANTY OF ANY KIND,
# EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE IMPLIED
# WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A PARTICULAR PURPOSE.
#
#
# OVERVIEW
#
# Bloaty is a static memory profiling tool which measures the size utilization
# of a target binary and outputs the result to a file.
#
# USAGE
#
# swift_add_bloaty(<target>
# [OPTIONS]
# [LOG_TOTAL_MEMORY]
# [LOG_TOTAL_MEMORY_OPTIONS arg1 arg2 ...]
# [WORKING_DIRECTORY working_directory]
# [REPORT_DIRECTORY report_directory]
# )
#
# Call this function to create a new cmake target which runs the `target`'s
# executable binary with bloaty applied. All targets created with
# `swift_add_bloaty` can be invoked by calling the common target 'do-all-bloaty'.
#
# BLOATY OPTIONS
#
# * SEGMENTS: Outputs what the run-time loader uses to determine what
# parts of the binary needs to be loaded/mapped into memory.
# * SECTIONS: Outputs the binary in finer details structured in different sections.
# * SYMBOLS: Outputs individual functions or variables.
# * COMPILEUNITS: Outputs what compile unit (and corresponding source file)
# each bit of the binary came from.
#
# * NUM: Set how many rows to show per level before collapsing into '[other]'.
# Set to '0' for unlimited, default: '20'.
#
# * SORT:
# vm - Sort the vm size column from largest to smallest and tells you
# how much space the binary will take when loaded into memory.
# file - Sort the file size column from largest to smallest and tells you
# how much space the binary is taking on disk.
# both - Default, sorts by max(vm, file).
#
# LOG_TOTAL_MEMORY enables the creation of a log file where the output from Bloaty
# is parsed and the total number of allocated static memory (VM size) gets inserted.
#
# LOG_TOTAL_MEMORY_OPTIONS enables additional options when logging of total memory
# is enabled.
# * --input_file: Set equal to a file path where an output file from Bloaty
# exists. [Default: `${report_directory}/${output_file}`]
# * --output_file: Set equal to a file path where the log file should be written.
# [Default: `${report_directory}/../memory_log.txt`]
# * --message: Add a custom message that gets concatenated with the reported
# total memory. [Default: `"Static memory usage:"`]
#
# WORKING_DIRECTORY changes the execution directory for the tool from the default
# folder `${CMAKE_CURRENT_BINARY_DIR}` to the given argument. For instance, if
# a user wants to utilize files located in a specific folder.
#
# REPORT_DIRECTORY changes the output directory for the tool from the default folder
# `${CMAKE_BINARY_DIR}/profiling/bloaty-reports` to the given argument.
# Example, using argument `/tmp`, outputs the result to `/tmp`.
#
# NOTE
#
# * A cmake option is available to control whether targets should be built,
# with the name ${PROJECT_NAME}_ENABLE_PROFILING.
#
# Running
#
# cmake -D<project>_ENABLE_PROFILING=ON ..
#
# will explicitly enable these targets from the command line at configure time.
#
option(${PROJECT_NAME}_ENABLE_PROFILING "Builds targets with profiling applied" OFF)
find_package(Bloaty)
if (NOT Bloaty_FOUND AND ${PROJECT_NAME}_ENABLE_PROFILING)
message(STATUS "Bloaty is not installed on system, will fetch content from source")
cmake_minimum_required(VERSION 3.14.0)
include(FetchContent)
FetchContent_Declare(
bloaty
GIT_REPOSITORY https://github.com/google/bloaty.git
GIT_TAG v1.1
GIT_SHALLOW TRUE
)
FetchContent_MakeAvailable(bloaty)
endif()
macro(eval_bloaty_target target)
if (NOT TARGET ${target})
message(FATAL_ERROR "Specified target \"${target}\" does not exist")
endif()
get_target_property(target_type ${target} TYPE)
if (NOT target_type STREQUAL EXECUTABLE)
message(FATAL_ERROR "Specified target \"${target}\" must be an executable type to register for profiling with Bloaty")
endif()
if (NOT (${PROJECT_NAME} STREQUAL ${CMAKE_PROJECT_NAME} AND
${PROJECT_NAME}_ENABLE_PROFILING))
return()
endif()
if (NOT TARGET do-all-memory-profiling)
add_custom_target(do-all-memory-profiling)
endif()
endmacro()
function(swift_add_bloaty target)
eval_bloaty_target(${target})
set(argOption SEGMENTS SECTIONS SYMBOLS COMPILEUNITS LOG_TOTAL_MEMORY)
set(argSingle NUM SORT WORKING_DIRECTORY REPORT_DIRECTORY)
set(argMulti LOG_TOTAL_MEMORY_OPTIONS)
cmake_parse_arguments(x "${argOption}" "${argSingle}" "${argMulti}" ${ARGN})
if (x_UNPARSED_ARGUMENTS)
message(FATAL_ERROR "Unparsed arguments ${x_UNPARSED_ARGUMENTS}")
endif()
set(target_name bloaty-${target})
set(output_file ${target_name}.txt)
set(working_directory ${CMAKE_CURRENT_BINARY_DIR})
if (x_WORKING_DIRECTORY)
set(working_directory ${x_WORKING_DIRECTORY})
endif()
set(report_directory ${CMAKE_BINARY_DIR}/profiling/bloaty-reports)
if (x_REPORT_DIRECTORY)
set(report_directory ${x_REPORT_DIRECTORY})
endif()
unset(resource_options)
if (x_SEGMENTS)
list(APPEND resource_options segments)
endif()
if (x_SECTIONS)
list(APPEND resource_options sections)
endif()
if (x_SYMBOLS)
list(APPEND resource_options symbols)
endif()
if (x_COMPILEUNITS)
list(APPEND resource_options compileunits)
endif()
if (DEFINED resource_options)
string(REPLACE ";" "," resource_options "${resource_options}")
set(resource_options -d ${resource_options})
endif()
if (DEFINED x_NUM)
list(APPEND resource_options -n ${x_NUM})
endif()
if (x_SORT)
list(APPEND resource_options -s ${x_SORT})
endif()
if (NOT Bloaty_FOUND)
add_custom_target(${target_name}
COMMAND $(MAKE) --directory=${bloaty_BINARY_DIR}
COMMENT "Bloaty is running on ${target}\ (output: \"${report_directory}/${output_file}\")"
COMMAND ${CMAKE_COMMAND} -E make_directory ${report_directory}
COMMAND ${CMAKE_COMMAND} -E echo \"bloaty with options: ${resource_options}\" > ${report_directory}/${output_file}
COMMAND ${bloaty_BINARY_DIR}/bloaty ${resource_options} $<TARGET_FILE:${target}> >> ${report_directory}/${output_file}
WORKING_DIRECTORY ${working_directory}
DEPENDS ${target}
)
else()
add_custom_target(${target_name}
COMMENT "Bloaty is running on ${target}\ (output: \"${report_directory}/${output_file}\")"
COMMAND ${CMAKE_COMMAND} -E make_directory ${report_directory}
COMMAND ${CMAKE_COMMAND} -E echo \"bloaty with options: ${resource_options}\" > ${report_directory}/${output_file}
COMMAND ${Bloaty_EXECUTABLE} ${resource_options} $<TARGET_FILE:${target}> >> ${report_directory}/${output_file}
WORKING_DIRECTORY ${working_directory}
DEPENDS ${target}
)
endif()
if (NOT TARGET do-all-bloaty)
add_custom_target(do-all-bloaty)
endif()
add_dependencies(do-all-bloaty ${target_name})
add_dependencies(do-all-memory-profiling do-all-bloaty)
if (x_LOG_TOTAL_MEMORY)
set(memory_input_file -i=${report_directory}/${output_file})
set(memory_output_file -o=${report_directory}/../memory_log.txt)
foreach (memory_option ${x_LOG_TOTAL_MEMORY_OPTIONS})
if (${memory_option} MATCHES "--input_file")
set(memory_input_file ${memory_option})
list(REMOVE_ITEM x_LOG_TOTAL_MEMORY_OPTIONS ${memory_option})
elseif (${memory_option} MATCHES "--output_file")
set(memory_output_file ${memory_option})
list(REMOVE_ITEM x_LOG_TOTAL_MEMORY_OPTIONS ${memory_option})
endif()
endforeach()
set(script_options ${x_LOG_TOTAL_MEMORY_OPTIONS})
list(APPEND script_options ${memory_input_file} ${memory_output_file})
add_custom_command(TARGET ${target_name} POST_BUILD
COMMAND python ${CMAKE_SOURCE_DIR}/cmake/common/scripts/parse_bloaty.py ${script_options}
)
endif()
endfunction()