diff --git a/src/header/stdio/scanf.rs b/src/header/stdio/scanf.rs index 22ade67f54314b82b8b600cc99c0515c1b62eee6..2e858d2039a5dc354ef7178a94a8b6d532189727 100644 --- a/src/header/stdio/scanf.rs +++ b/src/header/stdio/scanf.rs @@ -111,6 +111,10 @@ unsafe fn inner_scanf<R: Read>( } }; + // When an EOF occurs, eof is set, stuff is marked matched + // as usual, and finally it is returned + let mut eof = false; + let mut kind = IntKind::Int; loop { kind = match c { @@ -195,7 +199,7 @@ unsafe fn inner_scanf<R: Read>( } width = width.map(|w| w - 1); if !read!() { - break; + return Ok(matched); } if width.map(|w| w > 0).unwrap_or(true) && (byte == b'x' || byte == b'X') @@ -203,7 +207,7 @@ unsafe fn inner_scanf<R: Read>( radix = 16; width = width.map(|w| w - 1); if width.map(|w| w > 0).unwrap_or(true) && !read!() { - break; + return Ok(matched); } } continue; @@ -215,7 +219,7 @@ unsafe fn inner_scanf<R: Read>( n.push(byte as char); width = width.map(|w| w - 1); if width.map(|w| w > 0).unwrap_or(true) && !read!() { - break; + return Ok(matched); } } @@ -334,6 +338,7 @@ unsafe fn inner_scanf<R: Read>( } width = width.map(|w| w - 1); if width.map(|w| w > 0).unwrap_or(true) && !read!() { + eof = true; break; } } @@ -352,6 +357,7 @@ unsafe fn inner_scanf<R: Read>( } width = width.map(|w| w - 1); if width.map(|w| w > 0).unwrap_or(true) && !read!() { + eof = true; break; } } @@ -396,14 +402,18 @@ unsafe fn inner_scanf<R: Read>( let mut ptr: Option<*mut c_char> = if ignore { None } else { Some(ap.get()) }; - while width.map(|w| w > 0).unwrap_or(true) && !invert == matches.contains(&byte) - { + // While we haven't used up all the width, and it matches + while width.map(|w| w > 0).unwrap_or(true) && !invert == matches.contains(&byte) { if let Some(ref mut ptr) = ptr { **ptr = byte as c_char; *ptr = ptr.offset(1); } + // Decrease the width, and read a new character unless the width is 0 width = width.map(|w| w - 1); if width.map(|w| w > 0).unwrap_or(true) && !read!() { + // Reading a new character has failed, return after + // actually marking this as matched + eof = true; break; } } @@ -412,7 +422,7 @@ unsafe fn inner_scanf<R: Read>( *ptr = 0; matched += 1; } - } + }, b'n' => { if !ignore { *ap.get::<*mut c_int>() = count as c_int; @@ -421,6 +431,10 @@ unsafe fn inner_scanf<R: Read>( _ => return Err(-1), } + if eof { + return Ok(matched); + } + if width != Some(0) && c != b'n' { // It didn't hit the width, so an extra character was read and matched. // But this character did not match so let's reuse it. diff --git a/tests/expected/stdio/scanf.stdout b/tests/expected/stdio/scanf.stdout index 41de7bd48416da772dca4f2a35501865a798ffec..230c9d2a8ba823ac780e173313d569f8bbe48144 100644 --- a/tests/expected/stdio/scanf.stdout +++ b/tests/expected/stdio/scanf.stdout @@ -1,9 +1,14 @@ -2, { sa: 12, ia: 345, ib: 0, ic: 0, fa: 0.000000, da: 0.000000, ptr: (nil), char: a, string: } -3, { sa: 12, ia: 18, ib: 837, ic: 8, fa: 0.000000, da: 0.000000, ptr: (nil), char: a, string: } -2, { sa: 12, ia: 18, ib: 837, ic: 8, fa: 0.100000, da: 0.200000, ptr: (nil), char: a, string: } -1, { sa: 12, ia: 18, ib: 837, ic: 8, fa: 0.100000, da: 0.200000, ptr: 0xabcdef, char: a, string: } -1, { sa: 12, ia: 18, ib: 837, ic: 8, fa: 0.100000, da: 0.200000, ptr: 0xabcdef, char: a, string: Hello } -1, { sa: 12, ia: 15, ib: 837, ic: 8, fa: 0.100000, da: 0.200000, ptr: 0xabcdef, char: a, string: Hello } -2, { sa: 12, ia: 15, ib: 837, ic: 8, fa: 0.100000, da: 0.200000, ptr: 0xabcdef, char: h, string: elllo } -1, { sa: 12, ia: 0, ib: 8, ic: 8, fa: 0.100000, da: 0.200000, ptr: 0xabcdef, char: h, string: elllo } -0, { sa: 12, ia: 0, ib: 8, ic: 8, fa: 0.100000, da: 0.200000, ptr: 0xabcdef, char: h, string: elllo } +1, { sa: 12, ia: 0, ib: 0, ic: 0, fa: 0.000000, da: 0.000000, ptr: (nil), char: a, string1: , string2: , string3: , string4: } +2, { sa: 12, ia: 18, ib: 837, ic: 0, fa: 0.000000, da: 0.000000, ptr: (nil), char: a, string1: , string2: , string3: , string4: } +1, { sa: 12, ia: 18, ib: 837, ic: 0, fa: 0.100000, da: 0.000000, ptr: (nil), char: a, string1: , string2: , string3: , string4: } +0, { sa: 12, ia: 18, ib: 837, ic: 0, fa: 0.100000, da: 0.000000, ptr: (nil), char: a, string1: , string2: , string3: , string4: } +1, { sa: 12, ia: 18, ib: 837, ic: 0, fa: 0.100000, da: 0.000000, ptr: (nil), char: a, string1: Hello, string2: , string3: , string4: } +1, { sa: 12, ia: 15, ib: 837, ic: 0, fa: 0.100000, da: 0.000000, ptr: (nil), char: a, string1: Hello, string2: , string3: , string4: } +2, { sa: 12, ia: 15, ib: 837, ic: 0, fa: 0.100000, da: 0.000000, ptr: (nil), char: h, string1: elllo, string2: , string3: , string4: } +1, { sa: 12, ia: 0, ib: 8, ic: 0, fa: 0.100000, da: 0.000000, ptr: (nil), char: h, string1: elllo, string2: , string3: , string4: } +0, { sa: 12, ia: 0, ib: 8, ic: 0, fa: 0.100000, da: 0.000000, ptr: (nil), char: h, string1: elllo, string2: , string3: , string4: } +4, { sa: 12, ia: 0, ib: 8, ic: 0, fa: 0.100000, da: 0.000000, ptr: (nil), char: h, string1: e, string2: o, string3: l, string4: d } +4, { sa: 12, ia: 0, ib: 8, ic: 0, fa: 0.100000, da: 0.000000, ptr: (nil), char: h, string1: a, string2: e, string3: f, string4: dddddd } +4, { sa: 12, ia: 0, ib: 8, ic: 0, fa: 0.100000, da: 0.000000, ptr: (nil), char: h, string1: a, string2: e, string3: f, string4: dddddd } +1, { sa: 12, ia: 0, ib: 8, ic: 0, fa: 0.100000, da: 0.000000, ptr: (nil), char: h, string1: testbbbb, string2: e, string3: f, string4: dddddd } +3 "https" "//" "redox-os.org" "" diff --git a/tests/stdio/scanf.c b/tests/stdio/scanf.c index 8978928cea14b93e0faa2b3c87c396d73b977b80..c9ed3f438abfd1204e64ce90eab24d1648cb663a 100644 --- a/tests/stdio/scanf.c +++ b/tests/stdio/scanf.c @@ -12,7 +12,10 @@ struct params { double da; int *ptr; char c; - char string[20]; + char string1[20]; + char string2[20]; + char string3[20]; + char string4[20]; }; void test(char* fmt_in, char* input, struct params *p, ...) { @@ -22,8 +25,8 @@ void test(char* fmt_in, char* input, struct params *p, ...) { va_end(args); printf( - "%d, { sa: %hhd, ia: %d, ib: %d, ic: %d, fa: %f, da: %lf, ptr: %p, char: %c, string: %s }\n", - ret, p->sa, p->ia, p->ib, p->ic, p->fa, p->da, p->ptr, p->c, p->string + "%d, { sa: %hhd, ia: %d, ib: %d, ic: %d, fa: %f, da: %lf, ptr: %p, char: %c, string1: %s, string2: %s, string3: %s, string4: %s }\n", + ret, p->sa, p->ia, p->ib, p->ic, p->fa, p->da, p->ptr, p->c, p->string1, p->string2, p->string3, p->string4 ); } @@ -34,9 +37,27 @@ int main(void) { test("%x %i %i", "12 0x345 010", &p, &p.ia, &p.ib, &p.ic); test("%f.%lf", "0.1.0.2", &p, &p.fa, &p.da); test("%p", "0xABCDEF", &p, &p.ptr); - test("%s", "Hello World", &p, &p.string); + test("%s", "Hello World", &p, &p.string1); test("%3i", "0xFF", &p, &p.ia); - test("%c%3c", "hello", &p, &p.c, &p.string); + test("%c%3c", "hello", &p, &p.c, &p.string1); test("test: %2i%n", "test: 0xFF", &p, &p.ia, &p.ib); test("hello world%%", "hello world%", &p); + test("h%1[ae]ll%1[^a] wor%1[^\n]%[d]", "hello world", &p, &p.string1, &p.string2, &p.string3, &p.string4); + test("h%1[ae]ll%1[^a] wor%1[^\n]%[d]", "halle worfdddddd", &p, &p.string1, &p.string2, &p.string3, &p.string4); + test("h%1[ae]ll%1[^a] wor%1[^\n]%[d]", "halle worfdddddd", &p, &p.string1, &p.string2, &p.string3, &p.string4); + test("%[^a]%[b]", "testbbbb", &p, &p.string1, &p.string2); + + + // Scanf stolen from the url parsing in curl + char protobuf[16]; + char slashbuf[4]; + char hostbuf[100]; + char pathbuf[100]; + + int ret = sscanf( + "https://redox-os.org", "%15[^\n/:]:%3[/]%[^\n/?#]%[^\n]", + &protobuf, &slashbuf, &hostbuf, &pathbuf + ); + + printf("%d \"%s\" \"%s\" \"%s\" \"%s\"\n", ret, &protobuf, &slashbuf, &hostbuf, &pathbuf); }