-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathcmake-integration-build.el
395 lines (291 loc) · 17.3 KB
/
cmake-integration-build.el
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
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
;;; cmake-integration-build.el --- Retrieve targets -*- lexical-binding: t -*-
;;; Commentary:
;;; Code:
(require 'cmake-integration-variables)
(require 'cmake-integration-core)
(require 'cmake-integration-configure)
(defun cmake-integration--adjust-build-preset ()
"Adjust the build preset when changing the configure preset.
This function is added to `cmake-integration-after-set-configure-preset-hook'."
(if cmake-integration-configure-preset
(let ((presets (cmake-integration-get-build-presets)))
;; Only change the build preset if there is excactly one
;; build preset for the conffigure preset
(if (= (length presets) 1)
(setq cmake-integration-build-preset (car presets))
(setq cmake-integration-build-preset nil)))
(setq cmake-integration-build-preset nil)))
(add-hook 'cmake-integration-after-set-configure-preset-hook 'cmake-integration--adjust-build-preset)
(defun cmake-integration-get-last-build-preset-name ()
"Get the `name' field of the last preset used for build."
(cmake-integration--get-preset-name cmake-integration-build-preset))
;;;###autoload (autoload 'cmake-integration-get-build-presets "cmake-integration")
(defun cmake-integration-get-build-presets (&optional configure-preset)
"Get the build presets associated with CONFIGURE-PRESET.
Get the build presets in both `CMakePresets.json' and
`CMakeUserPresets.json' files as well as in any included files whose
configure preset is CONFIGURE-PRESET. If CONFIGURE-PRESET is not
provided, then the value in the `cmake-integration-configure-preset'
variable will be used."
(cmake-integration-get-presets-of-type 'buildPresets configure-preset))
;;;###autoload (autoload 'cmake-integration-select-build-preset "cmake-integration")
(defun cmake-integration-select-build-preset ()
"Select a build preset for CMake."
(interactive)
(when (not cmake-integration-configure-preset)
(error "Please, select a configure preset first"))
(let ((all-presets (cmake-integration-get-build-presets)))
(setq cmake-integration-build-preset
(cmake-integration-select-preset all-presets "Build preset: "))))
(defun cmake-integration--create-target-fullname (target-name &optional config-name)
"Return target constructed from TARGET-NAME and CONFIG-NAME."
(if config-name
(concat target-name cmake-integration--multi-config-separator config-name)
target-name))
(defun cmake-integration--create-target (target-name &optional config-name)
"Create a simple target containing only its TARGET-NAME for config CONFIG-NAME.
This is only used for the `all', `clean', and `install' targets."
(let ((target-full-name (cmake-integration--create-target-fullname target-name config-name)))
(list target-full-name)))
(defun cmake-integration-delete-build-folder ()
"Delete the current build folder.
It's useful in case you changed the generator, since CMake would
complain in that case."
(interactive)
(delete-directory (cmake-integration-get-build-folder) t))
(defun check-if-build-folder-exists-and-throws-if-not ()
"Check that the build folder exists and throws an error if not."
(unless (file-exists-p (cmake-integration-get-build-folder))
(error "The build folder is missing. Please run either `cmake-integration-cmake-reconfigure' or
`cmake-integration-cmake-configure-with-preset' to configure the project")))
(defun cmake-integration--save-and-compile-no-completion (target &optional extra-args)
"Save the buffer and compile TARGET also passing EXTRA-ARGS.
See the documentation of `cmake-integration-get-build-command' for the
EXTRA-ARGS parameter."
(save-buffer 0)
(let ((compile-command (cmake-integration-get-build-command target extra-args)))
(compile compile-command)))
(defun cmake-integration--get-target-type-from-name (target-name all-targets)
"Get the type of the target with name TARGET-NAME from ALL-TARGETS.
ALL-TARGETS is an alist like the one returned by
`cmake-integration--get-targets-from-codemodel-json-file-2'."
(let ((target (alist-get target-name all-targets nil nil 'equal)))
;; (cmake-integration--get-target-type target)
(alist-get 'type target)))
(defun cmake-integration--target-annotation-function (target-name)
"Annotation function that takes a TARGET-NAME and return an annotation for it.
This is used in `cmake-integration--get-target-using-completions'
when completing a target name to generate an annotation for that
target, which is shown during the completions if you are using
the marginalia package, or in Emacs standard completion buffer."
(pcase (car (split-string target-name cmake-integration--multi-config-separator))
("clean" (concat (cmake-integration--get-annotation-initial-spaces target-name) "Clean all compiled targets"))
("all" (concat (cmake-integration--get-annotation-initial-spaces target-name) "Compile all targets"))
(_ (concat (cmake-integration--get-annotation-initial-spaces target-name) (cmake-integration--get-target-type-from-name target-name minibuffer-completion-table)))
))
(defun cmake-integration--get-target-using-completions (list-of-targets)
"Ask the user to choose one of the targets in LIST-OF-TARGETS using completions."
(let ((completion-extra-properties '(:annotation-function cmake-integration--target-annotation-function)))
(completing-read "Target: " list-of-targets nil t)))
(defun cmake-integration--get-all-targets (json-filename)
"Get all targets for completion specified in JSON-FILENAME.
Get the name of all targets for completion, respecting the value
of the `*-targets-during-completion' variables.
If a prefix argument is provided, then the value of
`cmake-integration-include-subproject-targets-during-completion'
will be ignored."
(let* ((include-subprojects (or current-prefix-arg cmake-integration-include-subproject-targets-during-completion))
(list-of-targets (if include-subprojects
(cmake-integration--get-targets-from-codemodel-json-file-2
json-filename)
(cmake-integration--get-targets-from-codemodel-json-file-2
json-filename
'cmake-integration--target-is-in-projectIndex0-p))))
;; Filter the list of targets
(cond
;; Do not include utility and library targets
((and cmake-integration-hide-utility-targets-during-completion
cmake-integration-hide-library-targets-during-completion)
(seq-filter #'(lambda (target) (and
(cmake-integration--target-is-not-utility-p target)
(cmake-integration--target-is-not-library-p target)))
list-of-targets))
;; Do not include only utility targets
(cmake-integration-hide-utility-targets-during-completion
;; Do not include utility targets
(seq-filter 'cmake-integration--target-is-not-utility-p list-of-targets))
; Do not include only library targets
(cmake-integration-hide-library-targets-during-completion
(seq-filter 'cmake-integration--target-is-not-library-p list-of-targets))
;; Include all targets
(t list-of-targets))))
(defun cmake-integration--select-build-target ()
"Ask for a target to build and return the target name."
;; If the build folder is missing we should stop with an error
(check-if-build-folder-exists-and-throws-if-not)
(if-let* ((json-filename (cmake-integration--get-codemodel-reply-json-filename))
(list-of-targets (cmake-integration--get-all-targets json-filename))
(target (cmake-integration--get-target-using-completions list-of-targets)))
(setq cmake-integration-current-target target)
;; If `json-filename' is nil that means we could not find the
;; CMake reply with the file API, which means the query file is
;; missing.
(display-warning 'cmake-integration "Could not find list of targets due to CMake file API file
missing. Please run either `cmake-integration-cmake-reconfigure' or
`cmake-integration-cmake-configure-with-preset'.")
(setq cmake-integration-current-target nil)))
;;;###autoload (autoload 'cmake-integration-save-and-compile "cmake-integration")
(defun cmake-integration-save-and-compile ()
"Ask for a target name and compile it.
A list of target names is obtained from the project using CMake's
file API and completion is used to choose the desired target. If
that is not possible, ask for the target name without
completions."
(interactive)
;; Ask the user for a target and set the
;; cmake-integration-current-target variable with the chosen target
;; name
(cmake-integration--select-build-target)
(cmake-integration--save-and-compile-no-completion cmake-integration-current-target))
;;;###autoload (autoload 'cmake-integration-save-and-compile-last-target "cmake-integration")
(defun cmake-integration-save-and-compile-last-target (&optional extra-args)
"Recompile the last target that was compiled (or `all') also passing EXTRA-ARGS.
See the documentation of `cmake-integration-get-build-command' for the
EXTRA-ARGS parameter."
(interactive)
(cmake-integration--save-and-compile-no-completion
(or cmake-integration-current-target "all")
extra-args))
(defun cmake-integration-get-build-command (target &optional extra-args)
"Get the command to compile target TARGET passing EXTRA-ARGS to cmake.
EXTRA-ARGS is a list of strings, which will be joined with a space as
separation and then passed to cmake command to build the target."
(pcase-let* ((`(,target-name ,config-name)
(split-string target cmake-integration--multi-config-separator))
(project-root (cmake-integration--get-project-root-folder))
(preset-arg-or-build-folder (if cmake-integration-build-preset
(format "--preset %s" (cmake-integration-get-last-build-preset-name))
(cmake-integration--get-build-folder-relative-to-project)))
(build-args extra-args))
;; Add configuration argument if available
(when config-name (push (format "--config %s" config-name) build-args))
;; Add the target
(push (format "--target %s" target-name) build-args)
;; Add build folder or preset part
(push preset-arg-or-build-folder build-args)
(format "cd %s && cmake --build %s" project-root (string-join build-args " "))))
;; See CMake file API documentation for what projectIndex is
;; https://cmake.org/cmake/help/latest/manual/cmake-file-api.7.html
(defun cmake-integration--target-is-in-projectIndex0-p (target)
"Return t if the projectIndex field of TARGET is 0."
(eq (alist-get 'projectIndex target) 0))
(defun cmake-integration--target-is-not-utility-p (target)
"Return t if TARGET type is not `UTILITY'."
(not (equal (alist-get 'type target) "UTILITY")))
(defun cmake-integration--target-is-not-library-p (target)
"Return t if TARGET type is not `UTILITY'."
(let ((type (alist-get 'type target)))
(when type
(not (equal (car (cdr (split-string type "_"))) "LIBRARY")))))
(defun cmake-integration--get-targets-from-configuration (config &optional predicate)
"Get all targets in CONFIG that match PREDICATE."
(if predicate
(seq-filter predicate (alist-get 'targets config))
(alist-get 'targets config)))
(defun cmake-integration--get-target-name (target config-name)
"Get name for TARGET, including CONFIG-NAME (if not nil)."
(let* ((target-name (alist-get 'name target)))
(cmake-integration--create-target-fullname target-name config-name)))
(defun cmake-integration--add-name-to-target (target config-name)
"Add the target name to TARGET for CONFIG-NAME.
TARGET is a list of cons cells, including one with `name` field. This
function will extract the value in the name field and prepend it to the
list."
(let ((target-name (cmake-integration--get-target-name target config-name)))
(cons target-name target)))
(defun cmake-integration--get-prepared-targets-from-configuration (config config-name predicate)
"Get all targets in CONFIG with name CONFIG-NAME that match PREDICATE.
The implicit targets `all', `clean' and optional `install' targets will
be returned as well.
CONFIG-NAME is non-nil only when using ninja multi-config generator,
where we have more than one configuration."
(let ((install-rule? (cl-some (lambda (dir) (alist-get 'hasInstallRule dir))
(alist-get 'directories config)))
(targets-in-config (cmake-integration--get-targets-from-configuration config predicate))
(targets-and-name))
(setq targets-and-name (mapcar (lambda (target-info) (cmake-integration--add-name-to-target target-info config-name))
targets-in-config))
;; Add implicit 'all', 'clean' and optional 'install' targets
(cmake-integration--add-all-clean-install-targets
targets-and-name
config-name
install-rule?)))
(defun cmake-integration--add-all-clean-install-targets (targets config-name has-install-rule)
"Return TARGETS with extra `all', `clean' and `install' for CONFIG-NAME.
The `install' target is only included if HAS-INSTALL-RULE is true."
(let ((all-target (list (cmake-integration--create-target "all" config-name)))
(clean-target (list (cmake-integration--create-target "clean" config-name)))
(install-target (when has-install-rule (list (cmake-integration--create-target "install" config-name)))))
(nconc all-target clean-target install-target targets)))
(defun cmake-integration--get-targets-from-codemodel-json-file (&optional json-filename predicate)
"Return the targets found in JSON-FILENAME that match PREDICATE.
Return an alist of (target-name . target-info) elements for
targets found in JSON-FILENAME.
JSON-FILENAME must be a CMake API codemodel file. If is not provided,
`cmake-integration--get-codemodel-reply-json-filename' is used.
The returned alist includes:
- Explicit targets from the JSON file.
- Implicit targets: `all', `clean', and `install' (if applicable).
Each entry maps `target-name` to `target-info`."
(let* ((json-filename (or json-filename
;; If json-filename was not provided, get it from
;; 'cmake-integration--get-codemodel-reply-json-filename'.
(cmake-integration--get-codemodel-reply-json-filename)))
(all-config (alist-get 'configurations (json-read-file json-filename))))
;; The result of the nested `mapcar's below is a list of list of alists.
;; What we need a list of alists so remove one level and combine the next
;; level lists (of alists) of into a single list (of alists).
(apply #'nconc
;; process all-config vector
(mapcar (lambda (config)
;; config-name is non-nil only when using ninja
;; multi-config generator, where we have more
;; than one configuration
(let ((config-name (and (> (length all-config) 1)
(alist-get 'name config))))
(cmake-integration--get-prepared-targets-from-configuration config config-name predicate)))
;; Sequence mapped in the mapcar
all-config))))
(defun cmake-integration--add-type-field-to-target (target)
"Add a `type' field to a TARGET.
This will modify TARGET.
TARGET is a list containing the target name followend by many cons, with
each cons having some information about the TARGET. Particularly, TARGET
has a cons cell with a `jsonFile' car and a `\"someJsonFile.json\"'
filename. This json file will be read to extract the type that should be
added to TARGET."
;; ("target-name" (directoryIndex . 2) (id . "Continuous::@a44f0ac069e85531cdee") (jsonFile . "target-Continuous-Debug-32dbfb99931b31bc9c8f.json") (name . #1#) (projectIndex . 0))
(let ((target-name (car (split-string (car target) cmake-integration--multi-config-separator))))
(unless (or (equal target-name "all") (equal target-name "clean") (equal target-name "install"))
(let* ((json-file (alist-get 'jsonFile target))
(json-full-filename (file-name-concat (cmake-integration--get-reply-folder) json-file))
(target-json-data (json-read-file json-full-filename)))
;; Set the value from the json data to `type' field
(setf (alist-get 'type (cdr target)) (alist-get 'type target-json-data))))))
(defun cmake-integration--get-targets-from-codemodel-json-file-2 (&optional json-filename predicate)
"Return the targets found in JSON-FILENAME that respect PREDICATE.
This function is the same as
`cmake-integration--get-targets-from-codemodel-json-file',
with the exception that it adds the type of each target to a
`type' field in the target. The main use for this information is
during completion of target names, where this type information is
shown as an annotation."
;; Start with the list of targets returned by
;; `cmake-integration--get-targets-from-codemodel-json-file',
;; then loop over each target to add the type information, skipping
;; the "all", "clean" and "install" targets.
(let ((list-of-targets (cmake-integration--get-targets-from-codemodel-json-file json-filename predicate)))
(mapc 'cmake-integration--add-type-field-to-target list-of-targets)
list-of-targets))
(provide 'cmake-integration-build)
;;; cmake-integration-build.el ends here