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
|
/* NSC -- new Scala compiler
* Copyright 2005-2009 LAMP/EPFL
* @author Martin Odersky
*/
// $Id$
package scala.tools.nsc
import java.io.File
import java.lang.System
class Settings(error: String => Unit) {
def this() = this(Console.println)
private var allsettings: List[Setting] = List()
protected def getProperty(name: String): String =
if (System.getProperty(name) != "")
System.getProperty(name)
else null
protected val classpathDefault =
if (System.getProperty("env.classpath") ne null)
alternatePath(
getProperty("env.classpath"),
".")
else getProperty("java.class.path")
protected val bootclasspathDefault =
alternatePath(
concatPath(
getProperty("sun.boot.class.path"),
guessedScalaBootClassPath),
"")
protected val extdirsDefault =
alternatePath(
concatPath(
getProperty("java.ext.dirs"),
guessedScalaExtDirs),
"")
protected val pluginsDirDefault =
if (Properties.scalaHome == null)
""
else
new File(
new File(
new File(Properties.scalaHome, "misc"),
"scala-devel"),
"plugins").getAbsolutePath
protected def alternatePath(p1: String, p2: => String) =
if (p1 ne null) p1 else p2
protected def concatPath(p1: String, p2: String) =
if ((p1 ne null) && (p2 ne null)) p1 + File.pathSeparator + p2
else if (p1 ne null) p1
else p2
private def guessedScalaBootClassPath = {
val scalaHome = Properties.scalaHome
if (scalaHome ne null) {
val guessJar = new File(new File(new File(scalaHome), "lib"), "scala-library.jar")
if (guessJar.isFile()) guessJar.getPath()
else {
val guessDir = new File(new File(new File(scalaHome), "classes"), "library")
if (guessDir.isDirectory()) guessDir.getPath() else null
}
} else null
}
private def guessedScalaExtDirs = {
val scalaHome = Properties.scalaHome
if (scalaHome ne null) {
val guess = new File(new File(scalaHome), "lib")
if (guess.isDirectory()) guess.getPath else null
} else null
}
val debuginfo = new DebugSetting ("-g", "Specify level of generated debugging info", List("none", "source", "line", "vars", "notailcalls"), "vars", "vars")
val nowarnings = BooleanSetting ("-nowarn", "Generate no warnings").hideToIDE
val verbose = BooleanSetting ("-verbose", "Output messages about what the compiler is doing").hideToIDE
val deprecation = BooleanSetting ("-deprecation", "Output source locations where deprecated APIs are used").hideToIDE
val unchecked = BooleanSetting ("-unchecked", "Enable detailed unchecked warnings").hideToIDE
val classpath = (new StringSetting ("-classpath", "path", "Specify where to find user class files", classpathDefault) { override val abbreviation = "-cp" }).hideToIDE
val sourcepath = StringSetting ("-sourcepath", "path", "Specify where to find input source files", "").hideToIDE
val bootclasspath = StringSetting ("-bootclasspath", "path", "Override location of bootstrap class files", bootclasspathDefault).hideToIDE
val extdirs = StringSetting ("-extdirs", "dirs", "Override location of installed extensions", extdirsDefault).hideToIDE
val outdir = StringSetting ("-d", "directory", "Specify where to place generated class files", ".").hideToIDE
val encoding = StringSetting ("-encoding", "encoding", "Specify character encoding used by source files", Properties.encodingString).hideToIDE
val target = ChoiceSetting ("-target", "Specify for which target object files should be built", List("jvm-1.5", "jvm-1.4", "msil"), "jvm-1.5")
val printLate = BooleanSetting ("-print", "Print program with all Scala-specific features removed").hideToIDE
val XO = BooleanSetting ("-optimise", "Generates faster bytecode by applying optimisations to the program")
val explaintypes = BooleanSetting ("-explaintypes", "Explain type errors in more detail").hideToIDE
val uniqid = BooleanSetting ("-uniqid", "Print identifiers with unique names for debugging").hideToIDE
val version = BooleanSetting ("-version", "Print product version and exit").hideToIDE
val help = BooleanSetting ("-help", "Print a synopsis of standard options").hideToIDE
val Xhelp = BooleanSetting ("-X", "Print a synopsis of advanced options").hideToIDE
val argfiles = BooleanSetting ("@<file>", "A text file containing compiler arguments (options and source files)") // only for the help message
val assemname = StringSetting ("-Xassem", "file", "Name of the output assembly (only relevant with -target:msil)", "").dependsOn(target, "msil").hideToIDE
val assemrefs = StringSetting ("-Xassem-path", "path", "List of assemblies referenced by the program (only relevant with -target:msil)", ".").dependsOn(target, "msil").hideToIDE
val Xchecknull = BooleanSetting ("-Xcheck-null", "Emit warning on selection of nullable reference")
val Xwarninit = BooleanSetting ("-Xwarninit", "Warn about possible changes in initialization semantics")
val checkInit = BooleanSetting ("-Xcheckinit", "Add runtime checks on field accessors. Uninitialized accesses result in an exception being thrown.")
val noassertions = BooleanSetting ("-Xdisable-assertions", "Generate no assertions and assumptions")
val Xexperimental = BooleanSetting ("-Xexperimental", "Enable experimental extensions")
val XlogImplicits = BooleanSetting ("-Xlog-implicits", "Show more info on why some implicits are not applicable")
val Xnojline = new BooleanSetting("-Xnojline", "Do not use JLine for editing").hideToIDE
val nouescape = BooleanSetting ("-Xno-uescape", "Disables handling of \\u unicode escapes")
val plugin = MultiStringSetting("-Xplugin", "file", "Load a plugin from a file")
val disable = MultiStringSetting("-Xplugin-disable", "plugin", "Disable a plugin")
val showPlugins = BooleanSetting ("-Xplugin-list", "Print a synopsis of loaded plugins").hideToIDE
val pluginOptions = new MultiStringSetting("-P", "plugin:opt", "Pass an option to a plugin") { override def helpSyntax = "-P:<plugin>:<opt>" }
val require = MultiStringSetting("-Xplugin-require", "plugin", "Abort unless a plugin is available")
val pluginsDir = StringSetting ("-Xpluginsdir", "path", "Location to find compiler plugins", pluginsDirDefault)
val print = PhasesSetting ("-Xprint", "Print out program after")
val Xprintpos = BooleanSetting ("-Xprint-pos", "Print tree positions (as offsets)").hideToIDE
val printtypes = BooleanSetting ("-Xprint-types", "Print tree types (debugging option)").hideToIDE
val prompt = BooleanSetting ("-Xprompt", "Display a prompt after each error (debugging option)").hideToIDE
val resident = BooleanSetting ("-Xresident", "Compiler stays resident, files to compile are read from standard input").hideToIDE
val Xshowcls = StringSetting ("-Xshow-class", "class", "Show class info", "").hideToIDE
val Xshowobj = StringSetting ("-Xshow-object", "object", "Show object info", "").hideToIDE
val showPhases = BooleanSetting ("-Xshow-phases", "Print a synopsis of compiler phases").hideToIDE
val genPhaseGraph = StringSetting ("-Xgenerate-phase-graph", "filename", "Generate the phase graphs (outputs .dot files) to filenameX.dot", "").hideToIDE
val sourceReader = StringSetting ("-Xsource-reader", "classname", "Specify a custom method for reading source files", "scala.tools.nsc.io.SourceReader").hideToIDE
// val migrate2_7_2 = BooleanSetting ("-Xmigrate-to-2.7.2", "Issue warning messages to help in migration to 2.7.2")
val future = BooleanSetting ("-Xfuture", "Turn on future language features")
val Yhelp = BooleanSetting ("-Y", "Print a synopsis of private options").hideToIDE
val browse = PhasesSetting ("-Ybrowse", "Browse the abstract syntax tree after")
val check = PhasesSetting ("-Ycheck", "Check the tree at the end of the given phase. Specify \"all\" to check all checkable phases")
val Xcloselim = BooleanSetting ("-Yclosure-elim", "Perform closure elimination")
val Xcodebase = StringSetting ("-Ycodebase", "codebase", "Specify the URL containing the Scala libraries", "").hideToIDE
val debug = BooleanSetting ("-Ydebug", "Output debugging messages").hideToIDE
val Xdce = BooleanSetting ("-Ydead-code", "Perform dead code elimination")
val Xdetach = BooleanSetting ("-Ydetach", "Perform detaching of remote closures")
// val doc = BooleanSetting ("-Ydoc", "Generate documentation").hideToIDE
val inline = BooleanSetting ("-Yinline", "Perform inlining when possible")
val Xlinearizer = ChoiceSetting ("-Ylinearizer", "Linearizer to use", List("normal", "dfs", "rpo", "dump"), "rpo")
val log = PhasesSetting ("-Ylog", "Log operations in")
val logAll = BooleanSetting ("-Ylog-all", "Log all operations").hideToIDE
val noimports = BooleanSetting ("-Yno-imports", "Compile without any implicit imports")
val nopredefs = BooleanSetting ("-Yno-predefs", "Compile without any implicit predefined values")
val Yrecursion = IntSetting ("-Yrecursion", "Recursion depth used when locking symbols", 0, Some(0), None).hideToIDE
val script = StringSetting ("-Xscript", "object", "Compile as a script, wrapping the code into object.main()", "").hideToIDE
val Xshowtrees = BooleanSetting ("-Yshow-trees", "Show detailed trees when used in connection with -print:phase").hideToIDE
val skip = PhasesSetting ("-Yskip", "Skip")
val Xsqueeze = ChoiceSetting ("-Ysqueeze", "if on, creates compact code in matching", List("on","on","off"), "on").hideToIDE
val statistics = BooleanSetting ("-Ystatistics", "Print compiler statistics").hideToIDE
val stop = PhasesSetting ("-Ystop", "Stop after phase")
val refinementMethodDispatch
= ChoiceSetting ("-Ystruct-dispatch", "Selects what dispatch method to use for structural refinement method calls", List("no-cache", "mono-cache", "poly-cache"), "poly-cache").hideToIDE
val Xwarndeadcode = BooleanSetting ("-Ywarn-dead-code", "Emit warnings for dead code")
val Ynogenericsig = BooleanSetting ("-Yno-generic-signatures", "Suppress generation of generic signatures for Java")
val XnoVarargsConversion = BooleanSetting("-Xno-varargs-conversion", "disable varags conversion")
val selfInAnnots = BooleanSetting ("-Yself-in-annots", "Include a \"self\" identifier inside of annotations")
val suppressVTWarn = BooleanSetting ("-Ysuppress-vt-typer-warnings", "Suppress warnings from the typer when testing the virtual class encoding, NOT FOR FINAL!")
/** A list of all settings */
def allSettings: List[Setting] = allsettings.reverse
/** Disable a setting */
def disable(s: Setting) = {
allsettings = allsettings filter (s !=)
}
override def equals(that: Any) = that match {
case s:Settings =>
assert(this.allSettings.length == s.allSettings.length)
List.forall2 (
this.allSettings,
s.allSettings
)(_==_)
case _ => false
}
def checkDependencies: Boolean = {
def hasValue(s: Setting, value: String): Boolean = s match {
case bs: BooleanSetting => bs.value
case ss: StringSetting => ss.value == value
case cs: ChoiceSetting => cs.value == value
case _ => "" == value
}
var ok = true
for (setting <- allsettings if !setting.dependency.isEmpty) {
val (dep, value) = setting.dependency.get
if (! (setting.isDefault || hasValue(dep, value))) {
error("incomplete option " + setting.name + " (requires " + dep.name + ")")
ok = false
}
}
ok
}
/** Try to add additional command line parameters. */
def parseParams(line: String, error: String => Nothing) {
var args =
if (line.trim() == "") Nil
else List.fromArray(line.trim().split(" ")).map(_.trim())
while (!args.isEmpty) {
val argsBuf = args
if (args.head startsWith "-") {
for (setting <- allSettings)
args = setting.tryToSet(args);
}
else error("Parameter '" + args.head + "' does not start with '-'.")
if (argsBuf eq args)
error("Parameter '" + args.head + "' is not recognised by Scalac.")
}
}
/** A base class for settings of all types.
* Subclasses each define a `value' field of the appropriate type.
*/
abstract class Setting(descr: String) {
/** The name of the option as written on the command line, '-' included. */
def name: String
/** If first arg defines this setting, consume it as well as all following
* args needed to define the setting. If this can be done without
* error, set value field and return suffix of args else
* issue error message and return the arguments unchanged.
* If first arg does not define this setting return args unchanged.
*/
def tryToSet(args: List[String]): List[String]
/** The syntax defining this setting in a help string */
def helpSyntax: String = name
/** A description of the purpose of this setting in a help string */
def helpDescription = descr
/** Return a list of strings that can be used to recreate
* the receiver's current setting.
*/
def unparse: List[String]
/** override if option should be hidden from IDE.
*/
var hiddenToIDE: Boolean = false
def hideToIDE: this.type = {
hiddenToIDE = true
this
}
def showToIDE: this.type = {
hiddenToIDE = false
this
}
protected var setByUser: Boolean = false
def isDefault: Boolean = !setByUser
protected[Settings] var dependency: Option[(Setting, String)] = None
def dependsOn(s: Setting, value: String): this.type = { dependency = Some((s, value)); this }
def dependsOn(s: Setting): this.type = dependsOn(s, "")
def isStandard: Boolean = !isAdvanced && !isPrivate && !(name eq "-Y")
def isAdvanced: Boolean =
(name startsWith "-X") && !(name eq "-X")
def isPrivate: Boolean =
(name == "-P") || ((name startsWith "-Y") && !(name eq "-Y"))
/*
def isDocOption: Boolean =
!dependency.isEmpty && dependency.get._1 == doc
*/
// initialization
allsettings = this :: allsettings
}
/** A setting represented by a positive integer */
case class IntSetting(name: String, descr: String, default: Int, min: Option[Int], max: Option[Int]) extends Setting(descr) {
// Validate that min and max are consistent
(min, max) match {
case (Some(i), Some(j)) => assert(i <= j)
case _ => ()
}
// Helper to validate an input
private def isInputValid(k: Int): Boolean =
(min, max) match {
case (Some(i), Some(j)) => (i <= k) && (k <= j)
case (Some(i), None) => (i <= k)
case (None, Some(j)) => (k <= j)
case _ => true
}
// Helper to generate a textual explaination of valid inputs
private def getValidText: String =
(min, max) match {
case (Some(i), Some(j)) => "must be between "+i+" and "+j
case (Some(i), None) => "must be greater than or equal to "+i
case (None, Some(j)) => "must be less than or equal to "+j
case _ => throw new Error("this should never be used")
}
// Ensure that the default value is actually valid
assert(isInputValid(default))
protected var v: Int = default
def errorMsg = error("invalid setting for -"+name+" "+getValidText)
def value: Int = this.v
def value_=(s: Int) {
if (!isInputValid(s)) errorMsg
setByUser = true;
this.v = s
}
def tryToSet(args: List[String]): List[String] = args match {
case n :: rest if (name == n) =>
if (rest.isEmpty) {
error("missing argument")
args
} else {
try {
value = rest.head.toInt
} catch {
case e: java.lang.NumberFormatException => errorMsg
}
rest.tail
}
case _ => args
}
def unparse: List[String] =
if (value == default) Nil else List(name, value.toString)
override def equals(that: Any) = that match {
case is:IntSetting => this.name == is.name && this.value == is.value
case _ => false
}
}
/** A setting represented by a boolean flag (false, unless set) */
case class BooleanSetting(name: String, descr: String) extends Setting(descr) {
protected var v: Boolean = false
def value: Boolean = this.v
def value_=(s: Boolean) { setByUser = true; this.v = s }
def tryToSet(args: List[String]): List[String] = args match {
case n :: rest if (n == name) => value = true; rest
case _ => args
}
def unparse: List[String] = if (value) List(name) else Nil
override def equals(that: Any) = that match {
case bs:BooleanSetting => this.name == bs.name && this.value == bs.value
case _ => false
}
}
/** A setting represented by a string, (`default' unless set) */
case class StringSetting(name: String, arg: String, descr: String, default: String)
extends Setting(descr) {
def abbreviation: String = null
protected var v: String = default
def value: String = this.v
def value_=(s: String) { setByUser = true; this.v = s }
def tryToSet(args: List[String]): List[String] = args match {
case n :: rest if (name == n || abbreviation == n) =>
if (rest.isEmpty) {
error("missing argument")
args
} else {
value = rest.head
rest.tail
}
case _ => args
}
override def helpSyntax = name + " <" + arg + ">"
def unparse: List[String] =
if (value == default) Nil else List(name, value)
override def equals(that: Any) = that match {
case ss:StringSetting => this.name == ss.name && this.value == ss.value
case _ => false
}
}
/** A setting that accumulates all strings supplied to it */
case class MultiStringSetting(name: String, arg: String, descr: String)
extends Setting(descr) {
hideToIDE
protected var v: List[String] = Nil
def value = v
def appendToValue(str: String) { v = v ::: List(str) }
protected val nameColon = name + ":"
def tryToSet(args: List[String]): List[String] = args match {
case arg :: rest if (arg startsWith nameColon) =>
val toadd = arg.substring(nameColon.length())
if (toadd.length == 0) {
error("empty argument to " + nameColon)
args
} else {
appendToValue(toadd)
rest
}
case opt :: arg :: rest if (opt == name) =>
appendToValue(arg)
rest
case _ => args
}
override def helpSyntax = name + ":<" + arg + ">"
def unparse: List[String] =
for (opt <- value) yield nameColon+opt
override def equals(that: Any) = that match {
case mss:MultiStringSetting =>
this.name == mss.name &&
this.value.length == mss.value.length &&
List.forall2(this.value.sort(_<_), mss.value.sort(_<_))(_==_)
case _ => false
}
}
/** A setting represented by a string in a given set of <code>choices</code>,
* (<code>default</code> unless set).
*/
case class ChoiceSetting(name: String, descr: String, choices: List[String], default: String)
extends Setting(descr + choices.mkString(" (", ",", ")")) {
protected var v: String = default
def value: String = this.v
def value_=(s: String) { setByUser = true; this.v = s }
protected def argument: String = name.substring(1)
def tryToSet(args: List[String]): List[String] = args match {
case n :: rest if (n startsWith (name + ":")) =>
val choice = n.substring(name.length() + 1)
if (!(choices contains choice)) {
error(
if (choice == "") "missing " + argument
else "unknown " + argument + " '" + choice + "'")
args
} else {
value = choice
rest
}
case n :: choice :: rest if n == name => // alternative to be consistent with IDE
if (!(choices contains choice)) {
error(
if (choice == "") "missing " + argument
else "unknown " + argument + " '" + choice + "'")
args
} else {
value = choice
rest
}
case _ => args
}
override def helpSyntax = name + ":<" + argument + ">"
def unparse: List[String] =
if (value == default) Nil else List(name + ":" + value)
override def equals(that: Any) = that match {
case cs:ChoiceSetting => this.name == cs.name && this.value == cs.value
case _ => false
}
}
/** Same as ChoiceSetting but have a <code>level</code> int which tells the
* index of the selected choice. The <code>defaultEmpty</code> is used when
* this setting is used without specifying any of the available choices.
*/
class DebugSetting(name: String, descr: String, choices: List[String], default: String, defaultEmpty: String)
extends ChoiceSetting(name, descr, choices, default) {
def indexOf[a](xs: List[a], e: a): Option[Int] = xs match {
case y :: ys => if (e == y) Some(0) else indexOf(ys, e) match {
case Some(idx) => Some(1 + idx)
case None => None
}
case _ => None
}
var level: Int = indexOf(choices, default).get
override def value_=(choice: String) {
setByUser = true
this.v = choice
this.level = indexOf(choices, choice).get
}
override def tryToSet(args: List[String]): List[String] = args match {
case n :: rest if (n startsWith (name + ":")) =>
val choice = n.substring(name.length() + 1)
if (!(choices contains choice)) {
error(
if (choice == "") "missing " + argument
else "unknown " + argument + " '" + choice + "'")
args
} else {
value = choice
rest
}
case n :: rest if (n startsWith name) =>
value = defaultEmpty
rest
case _ => args
}
}
/** A setting represented by a list of strings which should be prefixes of
* phase names. This is not checked here, however.
* (the empty list, unless set)
*/
case class PhasesSetting(name: String, descr: String)
extends Setting(descr + " <phase>") { // (see -showphases)") {
hideToIDE
protected var v: List[String] = List()
def value: List[String] = this.v
def value_=(s: List[String]) { setByUser = true; this.v = s }
def tryToSet(args: List[String]): List[String] = args match {
case n :: rest if (n startsWith (name + ":")) =>
val phase = n.substring(name.length() + 1)
if (phase == "") {
error("missing phase")
args
} else {
value = value ::: List(phase)
rest
}
case _ => args
}
override def helpSyntax = name + ":<phase>"
def contains(phasename: String): Boolean =
value exists (str => phasename startsWith str)
def unparse: List[String] =
(value.foldLeft[List[String]]
(Nil)
((args, phase) =>
List(name + ":" + phase) ::: args))
override def equals(that: Any) = that match {
case ps:PhasesSetting =>
this.name == ps.name &&
this.value.length == ps.value.length &&
List.forall2(this.value.sort(_<_), ps.value.sort(_<_))(_==_)
case _ => false
}
}
}
|