-
Notifications
You must be signed in to change notification settings - Fork 90
/
coq-system.el
737 lines (644 loc) · 30 KB
/
coq-system.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
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
;;; coq-system.el --- common part of compilation feature -*- lexical-binding:t -*-
;; This file is part of Proof General.
;; Portions © Copyright 1994-2012 David Aspinall and University of Edinburgh
;; Portions © Copyright 2003-2018 Free Software Foundation, Inc.
;; Portions © Copyright 2001-2017 Pierre Courtieu
;; Portions © Copyright 2010, 2016 Erik Martin-Dorel
;; Portions © Copyright 2011-2013, 2016-2017 Hendrik Tews
;; Portions © Copyright 2015-2017 Clément Pit-Claudel
;; Authors: Hendrik Tews, Pierre Courtieu
;; Maintainer: Pierre.Courtieu<[email protected]>
;; License: GPL (GNU GENERAL PUBLIC LICENSE)
;;; Commentary:
;;
;; This file holds constants, options and some general functions for
;; setting coq command arguments. Some code is dedicated as a light
;; support for older versions of coq.
;;
;;; Code:
(require 'proof)
(require 'coq-mode) ;for coq-prog-name
(defvar coq-prog-args)
(defvar coq-debug)
;; Arbitrary arguments can already be passed through _CoqProject.
;; However this is not true for all assistants, so we don't modify the
;; (defpgcustom prog-args) declaration.
(defun coq--string-list-p (obj)
"Determine if OBJ is a list of strings."
(or (null obj) (and (consp obj) (stringp (car obj)) (coq--string-list-p (cdr obj)))))
(put 'coq-prog-args 'safe-local-variable #'coq--string-list-p)
(defcustom coq-prog-env nil
"List of environment settings to pass to Coq process.
On Windows you might need something like:
(setq coq-prog-env '(\"HOME=C:\\Program Files\\Coq\\\"))"
:group 'coq)
(defcustom coq-dependency-analyzer
(if (executable-find "coqdep") "coqdep"
(proof-locate-executable "coqdep" t '("C:/Program Files/Coq/bin")))
"Command to invoke coqdep."
:type 'string
:group 'coq)
(defcustom coq-compiler
(if (executable-find "coqc") "coqc"
(proof-locate-executable "coqc" t '("C:/Program Files/Coq/bin")))
"Command to invoke the coq compiler."
:type 'string
:group 'coq)
(defcustom coq-pinned-version nil
"Manual coq version override (rarely needed).
There should be no need to set this value unless you use old trunk versions from
the Coq github repository. For Coq versions with decent version numbers
Proof General detects the version automatically and adjusts itself. This
variable should contain nil or a version string."
:type 'string
:group 'coq)
(defvar coq-autodetected-version nil
"Version of Coq, as autodetected by `coq-autodetect-version'.")
(defvar coq-autodetected-help nil
"Options of Coq, as autodetected by `coq-autodetect-options'.
This variable contains the output of \"coqtop -help\"")
;;; error symbols
;; coq-unclassifiable-version
;;
;; This error is signaled with one data item -- the bad version string
(put 'coq-unclassifiable-version 'error-conditions
'(error coq-unclassifiable-version))
(put 'coq-unclassifiable-version 'error-message
"Proof General cannot classify your Coq version")
;;; version detection code
(defun coq-version (&optional may-recompute)
"Return the precomputed version of the current Coq toolchain.
With MAY-RECOMPUTE, try auto-detecting it if it isn't available."
(or coq-pinned-version
coq-autodetected-version
(when may-recompute
(coq-autodetect-version))))
(defun coq-show-version ()
"Show the version of Coq currently in use.
If it doesn't look right, try `coq-autodetect-version'."
(interactive)
(let ((version (coq-version nil)))
(if version
(message "Using Coq v%s" coq-autodetected-version)
(message "Coq version unknown at this time. Use `coq-autodetect-version' to autodetect."))))
(defun coq-callcoq (option &optional expectedretv)
"Call coqtop with the given OPTION and return the output.
The given option should make coqtop return immediately.
Optionally check the return code and return nil if the check
fails.
This function supports calling coqtop via tramp."
(let* ((default-directory
(if (file-accessible-directory-p default-directory)
default-directory
"/"))
(coq-command (shell-quote-argument (or coq-prog-name "coqtop"))))
(with-temp-buffer
;; Use `shell-command' via `find-file-name-handler' instead of
;; `process-line': when the buffer is running TRAMP, PG uses
;; `start-file-process', loading the binary from the remote server.
(let* ((shell-command-str (format "%s %s" coq-command (or option "")))
(fh (find-file-name-handler default-directory 'shell-command))
(retv (if fh (funcall fh 'shell-command shell-command-str (current-buffer))
(shell-command shell-command-str (current-buffer)))))
(if (or (not expectedretv) (equal retv expectedretv))
(buffer-string))))))
(defun coq-autodetect-version (&optional interactive-p)
"Detect and record the version of Coq currently in use.
Interactively (with INTERACTIVE-P), show that number."
(interactive '(t))
(setq coq-autodetected-version nil)
(let* ((str (coq-callcoq "-v" 0))
(mtch (and str (string-match "version \\([^ ]+\\)" str))))
(when mtch
(setq coq-autodetected-version (match-string 1 str))))
(when interactive-p (coq-show-version))
coq-autodetected-version)
(defun coq-autodetect-help (&optional interactive-p)
"Record the output of coqotp -help in `coq-autodetected-help'."
(interactive '(t))
(setq coq-autodetected-help (coq-callcoq "-help")))
(defun coq--version< (v1 v2)
"Compare Coq versions V1 and V2."
;; -snapshot is only supported by Emacs 24.5, not 24.3
(let ((version-regexp-alist `(("^[-_+ ]?snapshot$" . -4)
("^pl$" . 0)
,@version-regexp-alist)))
(version< v1 v2)))
(defcustom coq-pre-v85 nil
"Deprecated.
Use `coq-pinned-version' if you want to bypass the
version detection that Proof General does automatically."
:type 'boolean
:group 'coq)
(defun coq--pre-v85 ()
"Return non-nil if the auto-detected version of Coq is < 8.5.
Returns nil if the version can't be detected."
(let ((coq-version-to-use (or (coq-version t) "8.5")))
(condition-case err
(coq--version< coq-version-to-use "8.5snapshot")
(error
(cond
((equal (substring (cadr err) 0 15) "Invalid version")
(signal 'coq-unclassifiable-version coq-version-to-use))
(t (signal (car err) (cdr err))))))))
(defun coq--post-v86 ()
"Return t if the auto-detected version of Coq is >= 8.6.
Return nil if the version cannot be detected."
(let ((coq-version-to-use (or (coq-version t) "8.5")))
(condition-case err
(not (coq--version< coq-version-to-use "8.6"))
(error
(cond
((equal (substring (cadr err) 0 15) "Invalid version")
(signal 'coq-unclassifiable-version coq-version-to-use))
(t (signal (car err) (cdr err))))))))
(defun coq--post-v810 ()
"Return t if the auto-detected version of Coq is >= 8.10.
Return nil if the version cannot be detected."
(let ((coq-version-to-use (or (coq-version t) "8.9")))
(condition-case err
(not (coq--version< coq-version-to-use "8.10alpha"))
(error
(cond
((equal (substring (cadr err) 0 15) "Invalid version")
(signal 'coq-unclassifiable-version coq-version-to-use))
(t (signal (car err) (cdr err))))))))
(defun coq--post-v811 ()
"Return t if the auto-detected version of Coq is >= 8.11.
Return nil if the version cannot be detected."
(let ((coq-version-to-use (or (coq-version t) "8.10")))
(condition-case err
(not (coq--version< coq-version-to-use "8.11"))
(error
(cond
((equal (substring (cadr err) 0 15) "Invalid version")
(signal 'coq-unclassifiable-version coq-version-to-use))
(t (signal (car err) (cdr err))))))))
(defun coq--supports-topfile ()
"Return t if \"-topfile\" appears in coqtop options"
(string-match "-topfile" coq-autodetected-help)
)
(defcustom coq-use-makefile nil
"Whether to look for a Makefile to attempt to guess the command line.
Set to t if you want this feature, but note that it is deprecated."
:type 'string
:group 'coq)
(defcustom coq-www-home-page "http://coq.inria.fr/"
"Coq home page URL."
:type 'string
:group 'coq)
;;; utility functions for variables
(defun coq-load-path-safep (path)
"Check if PATH is a safe value for `coq-load-path'."
(and
(listp path)
(cl-every
(lambda (entry)
(or (stringp entry)
(and (listp entry)
(eq (car entry) 'rec)
(cl-every 'stringp (cdr entry))
(equal (length entry) 3))
(and (listp entry)
(eq (car entry) 'nonrec)
(cl-every 'stringp (cdr entry))
(equal (length entry) 3))
(and (listp entry)
(eq (car entry) 'recnoimport)
(cl-every 'stringp (cdr entry))
(equal (length entry) 3))
(and (listp entry)
(eq (car entry) 'ocamlimport)
(cl-every 'stringp (cdr entry))
(equal (length entry) 2))
(and (listp entry)
(cl-every 'stringp entry)
(equal (length entry) 2))))
path)))
(defcustom coq-load-path nil
"Non-standard coq library load path.
This list specifies the LoadPath extension for coqdep, coqc and
coqtop. Usually, the elements of this list are strings (for
\"-I\") or lists of two strings (for \"-R\" dir path and
\"-Q\" dir path).
The possible forms of elements of this list correspond to the 4
forms of include options (`-I' `-Q' and `-R'). An element can be
- A list of the form `(\\='ocamlimport dir)', specifying (in 8.5) a
directory to be added to ocaml path (`-I').
- A list of the form `(\\='rec dir path)' (where dir and path are
strings) specifying a directory to be recursively mapped to the
logical path `path' (`-R dir path').
- A list of the form `(\\='recnoimport dir path)' (where dir and
path are strings) specifying a directory to be recursively
mapped to the logical path `path' (`-Q dir path'), but not
imported (modules accessible for import with qualified names
only). Note that -Q dir \"\" has a special, nonrecursive meaning.
- A list of the form (8.4 only) `(\\='nonrec dir path)', specifying a
directory to be mapped to the logical path 'path' ('-I dir -as path').
For convenience the symbol `rec' can be omitted and entries of
the form `(dir path)' are interpreted as `(rec dir path)'.
A plain string maps to -Q ... \"\" in 8.5, and -I ... in 8.4.
Under normal circumstances this list does not need to
contain the coq standard library or \".\" for the current
directory (see `coq-load-path-include-current').
WARNING: if you use coq <= 8.4, the meaning of these options is
not the same (-I is for coq path)."
:type '(repeat (choice (string :tag "simple directory without path (-Q \"\") in 8.5, -I in 8.4")
(list :tag
"recursive directory with path (-R ... ...)"
(const rec)
(string :tag "directory")
(string :tag "log path"))
(list :tag
"recursive directory without recursive import with path (-Q ... ...)"
(const recnoimport)
(string :tag "directory")
(string :tag "log path"))
(list :tag
"compatibility for of -I (-I ... -as ... in coq<=8.4)"
(const nonrec)
(string :tag "directory")
(string :tag "log path"))
(list :tag
"ocaml path (-I ...)"
(const ocamlimport)
(string :tag "directory")
(string :tag "log path"))))
:safe 'coq-load-path-safep
:group 'coq-auto-compile)
(make-variable-buffer-local 'coq-load-path)
(defcustom coq-load-path-include-current t
"If t, let coqdep search the current directory too.
Should be t for normal users. If t, pass -Q dir \"\" to coqdep when
processing files in directory \"dir\" in addition to any entries in
`coq-load-path'.
This setting is only relevant with Coq < 8.5."
:type 'boolean
:safe 'booleanp
:group 'coq-auto-compile)
(make-obsolete-variable 'coq-load-path-include-current "Coq 8.5 does not need it" "4.3")
(defun coq-option-of-load-path-entry (entry &optional pre-v85)
"Translate a single ENTRY from `coq-load-path' into options.
See `coq-load-path' for the possible forms of ENTRY and to which
options they are translated. Use a non-nil PRE-V85 flag to
request compatibility handling of flags."
(if pre-v85
;; FIXME Which base directory do we expand against? Should the entries of
;; load-path just always be absolute?
;; NOTE we don't handle 'recnoimport in 8.4, and we don't handle 'nonrec
;; in 8.5.
(pcase entry
((or (and (pred stringp) dir) `(ocamlimport ,dir))
(list "-I" (expand-file-name dir)))
(`(nonrec ,dir ,alias)
(list "-I" (expand-file-name dir) "-as" alias))
((or `(rec ,dir ,alias) `(,dir ,alias))
(list "-R" (expand-file-name dir) alias)))
(pcase entry
((and (pred stringp) dir)
(list "-Q" (expand-file-name dir) ""))
(`(ocamlimport ,dir)
(list "-I" (expand-file-name dir)))
(`(recnoimport ,dir ,alias)
(list "-Q" (expand-file-name dir) alias))
((or `(rec ,dir ,alias) `(,dir ,alias))
(list "-R" (expand-file-name dir) alias)))))
(defun coq-include-options (loadpath &optional current-directory pre-v85)
"Build the base list of include options for coqc, coqdep and coqtop.
The options list includes all entries from argument LOADPATH
\(which should be `coq-load-path' of the buffer that invoked the
compilation) prefixed with suitable options and (for coq<8.5), if
`coq-load-path-include-current' is enabled, the directory base of
FILE. The resulting list is fresh for every call, callers can
append more arguments with `nconc'.
CURRENT-DIRECTORY should be an absolute directory name. It can be nil if
`coq-load-path-include-current' is nil.
A non-nil PRE-V85 flag requests compatibility handling of flags."
(unless (coq-load-path-safep loadpath)
(error "Invalid value in coq-load-path"))
(when (and pre-v85 coq-load-path-include-current)
(cl-assert current-directory)
(push current-directory loadpath))
(cl-loop for entry in loadpath
append (coq-option-of-load-path-entry entry pre-v85)))
(defun coq--options-test-roundtrip-1 (coq-project parsed pre-v85)
"Run a sanity check on COQ-PROJECT's PARSED options.
If PRE-V85 is non-nil, use compatibility mode."
(let* ((concatenated (apply #'append parsed))
(coq-load-path-include-current nil)
(extracted (coq--extract-load-path parsed nil))
(roundtrip (coq-include-options extracted nil pre-v85)))
(princ (format "[%s] with compatibility flag set to %S: " coq-project pre-v85))
(if (equal concatenated roundtrip)
(princ "OK\n")
(princ (format "Failed.\n:: Original: %S\n:: LoadPath: %S\n:: Roundtrip: %S\n"
concatenated extracted roundtrip)))))
(defun coq--options-test-roundtrip (coq-project &optional v85-only)
"Run a sanity check on COQ-PROJECT.
If V85-ONLY is non-nil, do not check the compatibility code."
(let ((parsed (coq--read-options-from-project-file coq-project)))
(coq--options-test-roundtrip-1 coq-project parsed nil)
(unless v85-only
(coq--options-test-roundtrip-1 coq-project parsed t))))
(defun coq--options-test-roundtrips ()
"Run sanity tests on coq-project parsing code.
More precisely, check that parsing and reprinting the include
options of a few coq-project files does the right thing."
(with-output-to-temp-buffer "*tests*"
(coq--options-test-roundtrip "-Q /test Test")
(coq--options-test-roundtrip "-Q /test \"\"")
(coq--options-test-roundtrip "-R /test Test")
(coq--options-test-roundtrip "-I /test")))
(defun coq-coqdep-prog-args (loadpath &optional current-directory pre-v85)
"Build a list of options for coqdep.
LOADPATH, CURRENT-DIRECTORY, PRE-V85: see `coq-include-options'."
(coq-include-options loadpath current-directory pre-v85))
(defun coq--clean-prog-args (args)
"Return ARGS without the entries added by `coq-coqtop-prog-args'.
Such entries are currently -emacs and -topfile."
(pcase args
((or `("-emacs" . ,rest)
`("-topfile" . (,(pred (apply-partially #'equal buffer-file-name)) . ,rest)))
(coq--clean-prog-args rest))
(`(,car . ,cdr)
(cons car (coq--clean-prog-args cdr)))
(_ args)))
(defun coq-coqc-prog-args (loadpath &optional current-directory pre-v85)
"Build a list of options for coqc.
LOADPATH, CURRENT-DIRECTORY, PRE-V85: see `coq-include-options'."
;; coqtop always adds the current directory to the LoadPath, so don't
;; include it in the -Q options.
(append (coq--clean-prog-args coq-prog-args)
(let ((coq-load-path-include-current nil)) ; Not needed in >=8.5beta3
(coq-coqdep-prog-args loadpath current-directory pre-v85))))
;;XXXXXXXXXXXXXX
;; coq-coqtop-prog-args is the user-set list of arguments to pass to
;; Coq process, see 'defpacustom prog-args' in pg-custom.el for
;; documentation.
;; Regular files should be of the form "valid_modulename.v" coq
;; accepts lots of things as letter, this is probably much stricter.
;; In practice it should be OK though, except maybe for non latin
;; characters.
(defun coq-regular-filename-p (s)
(let ((noext (file-name-base s)))
(string-match-p "\\`[[:alpha:]_][[:alnum:]_]*\\'" noext)))
(defun coq-coqtop-prog-args (loadpath &optional current-directory pre-v85)
;; coqtop always adds the current directory to the LoadPath, so don't
;; include it in the -Q options. This is not true for coqdep.
"Build a list of options for coqc.
LOADPATH, CURRENT-DIRECTORY, PRE-V85: see `coq-coqc-prog-args'."
(let ((topfile-supported (coq--supports-topfile)))
(append
;; it is better to inform coqtop of the name of the current module
;; using the -topfile option, so that coqtop understands references
;; to it. But don't insist if this would fail (because of wrong
;; file name): Some people want to script .v files without ever
;; compiling them.
(if (and topfile-supported buffer-file-name
(coq-regular-filename-p buffer-file-name))
(cons "-topfile" (cons buffer-file-name nil))
(if (and topfile-supported buffer-file-name)
(message "Warning: this Coq buffer is probably not compilable \
because of its name, no -topfile option set."))
nil)
(cons "-emacs" (coq-coqc-prog-args loadpath current-directory pre-v85)))))
(defun coq-prog-args ()
"Recompute `coq-load-path' before calling `coq-coqtop-prog-args'."
(coq-load-project-file)
(coq-autodetect-version)
(coq-autodetect-help)
(coq-coqtop-prog-args coq-load-path))
(defcustom coq-use-project-file t
"If t, when opening a Coq file read the dominating _CoqProject.
If t, when a Coq file is opened, Proof General will look for a
project file (see `coq-project-filename') somewhere in the
current directory or its parent directories. If there is one,
its contents are read and used to determine the arguments that
must be given to coqtop. In particular it sets the load
path (including the -R lib options) (see `coq-load-path')."
:type 'boolean
:safe 'booleanp
:group 'coq)
(defcustom coq-project-filename "_CoqProject"
"The name of coq project file.
The coq project file of a coq developpement (cf. Coq documentation on
\"makefile generation\") should contain the arguments given to
coq_makefile. In particular it contains the -I and -R
options (preferably one per line). If `coq-use-coqproject' is
t (default) the content of this file will be used by proofgeneral to
infer the `coq-load-path' and the `coq-prog-args' variables that set
the coqtop invocation by proofgeneral. This is now the recommended
way of configuring the coqtop invocation. Local file variables may
still be used to override the coq project file's configuration.
.dir-locals.el files also work and override project file settings."
:type 'string
:safe 'stringp
:group 'coq)
(defun coq-find-project-file ()
"Return '(buf alreadyopen) where buf is the buffer visiting coq project file.
alreadyopen is t if buffer already existed."
(when buffer-file-name
(let* ((projectfiledir (locate-dominating-file buffer-file-name coq-project-filename)))
(when projectfiledir
(let* ((projectfile (expand-file-name coq-project-filename projectfiledir))
;; we store this intermediate result to know if we have to kill
;; the coq project buffer at the end
(projectbufferalreadyopen (find-buffer-visiting projectfile))
(projectbuffer (or projectbufferalreadyopen
(find-file-noselect projectfile t t))))
(list projectbuffer projectbufferalreadyopen))))))
(defconst coq--project-file-separator "[\r\n[:space:]]+")
(defconst coq--makefile-switch-arities
'(("-R" . 2)
("-Q" . 2)
("-I" . 1)
("-arg" . 1)
("-opt" . 0)
("-byte" . 0)))
(defun coq--read-one-option-from-project-file (switch arity raw-args)
"Cons SWITCH with ARITY arguments from RAW-ARGS.
If ARITY is nil, return SWITCH."
(if arity
(let ((arguments
(condition-case-unless-debug nil
(cl-subseq raw-args 0 arity)
(warn "Invalid _CoqProject: not enough arguments for %S" switch))))
(cons switch arguments))
switch))
(defun coq--read-options-from-project-file (contents)
"Read options from CONTENTS of _CoqProject.
Returns a mixed list of option-value pairs and strings."
(let ((raw-args (split-string-and-unquote contents coq--project-file-separator))
(options nil))
(while raw-args
(let* ((switch (pop raw-args))
(arity (cdr (assoc switch coq--makefile-switch-arities))))
(push (coq--read-one-option-from-project-file switch arity raw-args) options)
(setq raw-args (nthcdr (or arity 0) raw-args))))
(nreverse options))) ; Order of options is important sometimes (Cf. #7980)
(defun coq--extract-prog-args (options)
"Extract coqtop arguments from _CoqProject options OPTIONS.
OPTIONS is a list or conses (switch . argument) and strings.
Note that the semantics of the -arg flags in coq project files
are weird: -arg \"a b\" means pass a and b, separately, to
coqtop."
(let ((args nil))
(dolist (opt options)
(pcase opt
((or "-byte" "-op")
(push opt args))
(`("-arg" ,concatenated-args)
(setq args
(append args
(split-string-and-unquote concatenated-args coq--project-file-separator))))))
(cons "-emacs" args)))
(defun coq--extract-load-path-1 (option base-directory)
"Convert one _CoqProject OPTION, relative to BASE-DIRECTORY."
(pcase option
(`("-I" ,path)
(list 'ocamlimport (expand-file-name path base-directory)))
(`("-R" ,path ,alias)
(list 'rec (expand-file-name path base-directory) alias))
(`("-Q" ,path ,alias)
(list 'recnoimport (expand-file-name path base-directory) alias))))
(defun coq--extract-load-path (options base-directory)
"Extract loadpath from _CoqProject options OPTIONS.
OPTIONS is a list or conses (switch . arguments) and strings.
Paths are taken relative to BASE-DIRECTORY."
(cl-loop for opt in options
for extracted = (coq--extract-load-path-1 opt base-directory)
when extracted collect extracted))
;; optional args allow to implement the precedence of dir/file local vars
(defun coq-load-project-file-with-avoid (&optional avoidargs avoidpath)
"Set `coq-prog-args' and `coq-load-path' from _CoqProject.
If AVOIDARGS or AVOIDPATH is set, do not set the corresponding
variable."
(pcase-let* ((`(,proj-file-buf ,no-kill) (coq-find-project-file)))
(if (not proj-file-buf)
(message "Coq project file not detected.")
(let* ((contents (with-current-buffer proj-file-buf (buffer-string)))
(options (coq--read-options-from-project-file contents))
(proj-file-name (buffer-file-name proj-file-buf))
(proj-file-dir (file-name-directory proj-file-name))
(proj-file-local-dir (or (file-remote-p proj-file-dir 'localname)
proj-file-dir)))
(unless avoidargs (setq coq-prog-args (coq--extract-prog-args options)))
(unless avoidpath (setq coq-load-path (coq--extract-load-path options proj-file-local-dir)))
(let ((msg
(cond
((and avoidpath avoidargs) "Coqtop args and load path")
(avoidpath "Coqtop load path")
(avoidargs "Coqtop args")
(t ""))))
(message
"Coq project file detected: %s%s." proj-file-name
(if (or avoidpath avoidargs)
(concat "\n(" msg " overridden by dir/file local values)")
"")))
(when coq-debug
(message "coq-prog-args: %S" coq-prog-args)
(message "coq-load-path: %S" coq-load-path))
(unless no-kill (kill-buffer proj-file-buf))))))
(defun coq-load-project-file ()
"Set `coq-prog-args' and `coq-load-path' according to _CoqProject file.
Obeys `coq-use-project-file'. Note that if a variable is already set
by dir/file local variables, this function will not override its value.
See `coq-project-filename' to change the name of the project file, and
`coq-use-project-file' to disable this feature."
(when coq-use-project-file
;; Let us reread dir/file local vars, in case the user mmodified them
(let* ((oldargs (assoc 'coq-prog-args file-local-variables-alist))
(oldpath (assoc 'coq-load-path file-local-variables-alist)))
(coq-load-project-file-with-avoid oldargs oldpath))))
(defun coq-load-project-file-rehack ()
"Reread file/dir local vars and call `coq-load-project-file'.
Does nothing if `coq-use-project-file' is nil."
(when coq-use-project-file
;; Let us reread dir/file local vars, in case the user mmodified them
(hack-local-variables)
;; Useless since coq-load-project-file is in hack-local-variables-hook:
;;(coq-load-project-file)
))
;; Since coq-project-filename can be set via .dir-locals.el or file variable,
;; we need to call coq-load-coq-project-file only *after* local variables are
;; set. But coq-mode-hook is called BEFORE local variables are read. Therefore
;; coq-load-coq-project-file is added to hack-local-variables-hook instead. To
;; avoid adding for other modes , the setting is performed inside
;; coq-mode-hook. This is described in www.emacswiki.org/emacs/LocalVariables
;; TODO: also read COQBIN somewhere?
;; Note: this does not need to be at a particular place in the hook, but we
;; need to make this hook local.
;; hack-local-variables-hook seems to hack local and dir local vars.
(add-hook 'coq-mode-hook
'(lambda ()
(add-hook 'hack-local-variables-hook
'coq-load-project-file
nil t)))
; detecting coqtop args should happen at the last moment before
; calling the process. In particular it should ahppen after that
; proof-prog-name-ask is performed, this hook is at the right place.
(add-hook 'proof-shell-before-process-hook
'(lambda ()
;; It seems coq-prog-name and proof-prog-name are not correctly linked
;; so let us make sure they are the same before computing options
(setq coq-prog-name proof-prog-name)
(setq coq-prog-args (coq-prog-args))))
;; smie's parenthesis blinking is too slow, let us have the default one back
(add-hook 'coq-mode-hook
'(lambda ()
(when (and (fboundp 'show-paren--default)
(boundp 'show-paren-data-function))
(setq show-paren-data-function 'show-paren--default))))
(defun coq-toggle-use-project-file ()
(interactive)
(setq coq-use-project-file (not coq-use-project-file))
(when coq-use-project-file (coq-load-project-file-rehack))
;; FIXME What should we do when disabling project file? since
;; local variables override project file anyway, reading them
;; again is useless. Let us do nothing.
;;(setq coq-load-path nil)
;;(setq coq-prog-args nil)
)
(provide 'coq-system)
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; To guess the command line options ;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; OBSOLETE, should take _CoqProject into account.
(defun coq-guess-command-line (filename)
"Guess the right command line options to compile FILENAME using `make -n'.
This function is obsolete, the recommended way of setting the coqtop
options is to use a _Coqproject file as described in coq
documentation. ProofGeneral reads this file and sets compilation
options according to its contents. See `coq-project-filename'. Per
file configuration may then be set using local file variables."
(if (local-variable-p 'coq-prog-name (current-buffer))
coq-prog-name
(let* ((dir (or (file-name-directory filename) "."))
(makedir
(cond
((file-exists-p (concat dir "Makefile")) ".")
;; ((file-exists-p (concat dir "../Makefile")) "..")
;; ((file-exists-p (concat dir "../../Makefile")) "../..")
(t nil))))
(if (and coq-use-makefile makedir)
(let*
;;TODO, add dir name when makefile is in .. or ../..
;;
;; da: FIXME: this code causes problems if the make
;; command fails. It should not be used by default
;; and should be made more robust.
;;
((compiled-file (concat (substring
filename 0
(string-match ".v$" filename)) ".vo"))
(command (shell-command-to-string
(concat "cd " dir ";"
"make -n -W " filename " " compiled-file
"| sed s/coqc/coqtop/"))))
(message command)
(setq coq-prog-args nil)
(concat
(substring command 0 (string-match " [^ ]*$" command))
"-emacs"))
coq-prog-name))))
;;; coq-system.el ends here