Skip to content

Commit a47cd50

Browse files
committedMar 25, 2011
Parse FP literals without coercing to int. This allows parsing 64-bit
floats. However, if someone writes a literal that can't be represented precisely in 64 bits, the front-end will accept it while the back-end will (presumably) complain.
1 parent e2d6475 commit a47cd50

File tree

4 files changed

+71
-39
lines changed

4 files changed

+71
-39
lines changed
 

‎Makefile.in

+3-3
Original file line numberDiff line numberDiff line change
@@ -572,7 +572,8 @@ docsnap: doc/rust.pdf
572572

573573
FLOAT_XFAILS := $(S)src/test/run-pass/float.rs \
574574
$(S)src/test/run-pass/float2.rs \
575-
$(S)src/test/run-pass/float-signature.rs
575+
$(S)src/test/run-pass/float-signature.rs \
576+
$(S)src/test/run-pass/floatlits.rs
576577

577578
# Temporarily xfail tests broken by the nominal-tags change.
578579

@@ -641,8 +642,7 @@ TEST_XFAILS_BOOT := $(TASK_XFAILS) \
641642
writing-through-read-alias.rs) \
642643
$(S)src/test/bench/shootout/nbody.rs
643644

644-
TEST_XFAILS_STAGE0 := $(FLOAT_XFAILS) \
645-
$(addprefix $(S)src/test/run-pass/, \
645+
TEST_XFAILS_STAGE0 := $(addprefix $(S)src/test/run-pass/, \
646646
acyclic-unwind.rs \
647647
alt-pattern-drop.rs \
648648
alt-type-simple.rs \

‎src/comp/front/lexer.rs

+59-36
Original file line numberDiff line numberDiff line change
@@ -318,47 +318,65 @@ impure fn consume_block_comment(reader rdr) {
318318
be consume_any_whitespace(rdr);
319319
}
320320

321-
impure fn scan_dec_digits(reader rdr) -> int {
322-
323-
auto c = rdr.curr();
321+
fn digits_to_string(str s) -> int {
324322

325323
let int accum_int = 0;
324+
let int i = 0;
326325

327-
while (is_dec_digit(c) || c == '_') {
328-
if (c != '_') {
329-
accum_int *= 10;
330-
accum_int += dec_digit_val(c);
331-
}
332-
rdr.bump();
333-
c = rdr.curr();
326+
for (u8 c in s) {
327+
accum_int *= 10;
328+
accum_int += dec_digit_val(c as char);
334329
}
335330

336331
ret accum_int;
337332
}
338333

339-
impure fn scan_exponent(reader rdr) -> option.t[int] {
334+
impure fn scan_exponent(reader rdr) -> option.t[str] {
340335
auto c = rdr.curr();
341-
auto sign = 1;
336+
auto res = "";
342337

343338
if (c == 'e' || c == 'E') {
339+
res += _str.from_bytes(vec(c as u8));
344340
rdr.bump();
345341
c = rdr.curr();
346-
if (c == '-') {
347-
sign = -1;
348-
rdr.bump();
349-
} else if (c == '+') {
342+
if (c == '-' || c == '+') {
343+
res += _str.from_bytes(vec(c as u8));
350344
rdr.bump();
351345
}
352346
auto exponent = scan_dec_digits(rdr);
353-
ret(some(sign * exponent));
347+
if (_str.byte_len(exponent) > 0u) {
348+
ret(some(res + exponent));
349+
}
350+
else {
351+
log ("scan_exponent: bad fp literal");
352+
fail;
353+
}
354354
}
355355
else {
356-
ret none[int];
356+
ret none[str];
357357
}
358358
}
359359

360+
impure fn scan_dec_digits(reader rdr) -> str {
361+
362+
auto c = rdr.curr();
363+
let str res = "";
364+
365+
while (is_dec_digit (c) || c == '_') {
366+
if (c != '_') {
367+
res += _str.from_bytes(vec(c as u8));
368+
}
369+
rdr.bump();
370+
c = rdr.curr();
371+
}
372+
373+
ret res;
374+
}
375+
360376
impure fn scan_number(mutable char c, reader rdr) -> token.token {
361377
auto accum_int = 0;
378+
let str dec_str = "";
379+
let bool is_dec_integer = false;
362380
auto n = rdr.next();
363381

364382
if (c == '0' && n == 'x') {
@@ -386,7 +404,12 @@ impure fn scan_number(mutable char c, reader rdr) -> token.token {
386404
c = rdr.curr();
387405
}
388406
} else {
389-
accum_int = scan_dec_digits(rdr);
407+
dec_str = scan_dec_digits(rdr);
408+
is_dec_integer = true;
409+
}
410+
411+
if (is_dec_integer) {
412+
accum_int = digits_to_string(dec_str);
390413
}
391414

392415
c = rdr.curr();
@@ -443,20 +466,19 @@ impure fn scan_number(mutable char c, reader rdr) -> token.token {
443466
}
444467
}
445468
c = rdr.curr();
469+
446470
if (c == '.') {
447471
// Parse a floating-point number.
448472
rdr.bump();
449-
auto accum_int1 = scan_dec_digits(rdr);
450-
auto base_str = _int.to_str(accum_int, 10u) + "."
451-
+ _int.to_str(accum_int1, 10u);
473+
auto dec_part = scan_dec_digits(rdr);
474+
auto float_str = dec_str + "." + dec_part;
452475
c = rdr.curr();
453-
auto exponent_str = "";
454-
let option.t[int] maybe_exponent = scan_exponent(rdr);
455-
alt(maybe_exponent) {
456-
case(some[int](?i)) {
457-
exponent_str = "e" + _int.to_str(i, 10u);
476+
auto exponent_str = scan_exponent(rdr);
477+
alt (exponent_str) {
478+
case (some[str](?s)) {
479+
float_str += s;
458480
}
459-
case(none[int]) {
481+
case (none[str]) {
460482
}
461483
}
462484

@@ -468,27 +490,28 @@ impure fn scan_number(mutable char c, reader rdr) -> token.token {
468490
if (c == '3' && n == '2') {
469491
rdr.bump(); rdr.bump();
470492
ret token.LIT_MACH_FLOAT(util.common.ty_f32,
471-
base_str + exponent_str);
493+
float_str);
472494
}
473495
else if (c == '6' && n == '4') {
474496
rdr.bump(); rdr.bump();
475497
ret token.LIT_MACH_FLOAT(util.common.ty_f64,
476-
base_str + exponent_str);
498+
float_str);
499+
/* FIXME: if this is out of range for either a 32-bit or
500+
64-bit float, it won't be noticed till the back-end */
477501
}
478502
}
479503
else {
480-
ret token.LIT_FLOAT(base_str + exponent_str);
504+
ret token.LIT_FLOAT(float_str);
481505
}
482506
}
483507

484508
auto maybe_exponent = scan_exponent(rdr);
485509
alt(maybe_exponent) {
486-
case(some[int](?i)) {
487-
ret token.LIT_FLOAT(_int.to_str(accum_int, 10u)
488-
+ "e" + _int.to_str(i, 10u));
510+
case(some[str](?s)) {
511+
ret token.LIT_FLOAT(dec_str + s);
489512
}
490-
case(none[int]) {
491-
ret token.LIT_INT(accum_int);
513+
case(none[str]) {
514+
ret token.LIT_INT(accum_int);
492515
}
493516
}
494517
}

‎src/rt/rustrt.def.in

+1
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,7 @@ upcall_grow_task
4444
upcall_join
4545
upcall_kill
4646
upcall_log_double
47+
upcall_log_float
4748
upcall_log_int
4849
upcall_log_str
4950
upcall_malloc

‎src/test/run-pass/floatlits.rs

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
fn main() {
2+
auto f = 4.999999999999;
3+
check (f > 4.90);
4+
check (f < 5.0);
5+
auto g = 4.90000000001e-10;
6+
check(g > 5e-11);
7+
check(g < 5e-9);
8+
}

0 commit comments

Comments
 (0)