Server IP : 85.214.239.14 / Your IP : 18.217.5.77 Web Server : Apache/2.4.62 (Debian) System : Linux h2886529.stratoserver.net 4.9.0 #1 SMP Tue Jan 9 19:45:01 MSK 2024 x86_64 User : www-data ( 33) PHP Version : 7.4.18 Disable Function : pcntl_alarm,pcntl_fork,pcntl_waitpid,pcntl_wait,pcntl_wifexited,pcntl_wifstopped,pcntl_wifsignaled,pcntl_wifcontinued,pcntl_wexitstatus,pcntl_wtermsig,pcntl_wstopsig,pcntl_signal,pcntl_signal_get_handler,pcntl_signal_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority,pcntl_async_signals,pcntl_unshare, MySQL : OFF | cURL : OFF | WGET : ON | Perl : ON | Python : ON | Sudo : ON | Pkexec : OFF Directory : /usr/share/doc/re2c/examples/rust/fill/ |
Upload File : |
/* Generated by re2c */ // re2rust $INPUT -o $OUTPUT use std::fs::File; use std::io::{Read, Write}; const YYMAXFILL: usize = 1; const BUFSIZE: usize = 4096; struct State { file: File, buf: [u8; BUFSIZE], lim: usize, cur: usize, mar: usize, tok: usize, eof: bool, } #[derive(PartialEq)] enum Fill { Ok, Eof, LongLexeme } fn fill(st: &mut State, need: usize) -> Fill { if st.eof { return Fill::Eof; } // Error: lexeme too long. In real life can reallocate a larger buffer. if st.tok < need { return Fill::LongLexeme; } // Shift buffer contents (discard everything up to the current token). st.buf.copy_within(st.tok..st.lim, 0); st.lim -= st.tok; st.cur -= st.tok; st.mar = st.mar.overflowing_sub(st.tok).0; // underflows if marker is unused st.tok = 0; // Fill free space at the end of buffer with new data from file. let n = match st.file.read(&mut st.buf[st.lim..BUFSIZE - YYMAXFILL]) { Ok(n) => n, Err(why) => panic!("cannot read from file: {}", why) }; st.lim += n; // If read zero characters, this is end of input => add zero padding // so that the lexer can access characters at the end of buffer. if n == 0 { st.eof = true; for i in 0..YYMAXFILL { st.buf[st.lim + i] = 0; } st.lim += YYMAXFILL; } return Fill::Ok; } fn lex(st: &mut State) -> isize { let mut count: isize = 0; 'lex: loop { st.tok = st.cur; { #[allow(unused_assignments)] let mut yych : u8 = 0; let mut yystate : usize = 0; 'yyl: loop { match yystate { 0 => { if st.lim - st.cur < 1 { if fill(st, 1) != Fill::Ok { return -1; } } yych = unsafe {*st.buf.get_unchecked(st.cur)}; st.cur += 1; match yych { 0x00 => { yystate = 1; continue 'yyl; } 0x20 => { yystate = 3; continue 'yyl; } 0x27 => { yystate = 5; continue 'yyl; } _ => { yystate = 2; continue 'yyl; } } } 1 => { // Check that it is the sentinel, not some unexpected null. return if st.tok == st.lim - YYMAXFILL { count } else { -1 } } 2 => { return -1; } 3 => { if st.lim - st.cur < 1 { if fill(st, 1) != Fill::Ok { return -1; } } yych = unsafe {*st.buf.get_unchecked(st.cur)}; match yych { 0x20 => { st.cur += 1; yystate = 3; continue 'yyl; } _ => { yystate = 4; continue 'yyl; } } } 4 => { continue 'lex; } 5 => { if st.lim - st.cur < 1 { if fill(st, 1) != Fill::Ok { return -1; } } yych = unsafe {*st.buf.get_unchecked(st.cur)}; st.cur += 1; match yych { 0x27 => { yystate = 6; continue 'yyl; } 0x5C => { yystate = 7; continue 'yyl; } _ => { yystate = 5; continue 'yyl; } } } 6 => { count += 1; continue 'lex; } 7 => { if st.lim - st.cur < 1 { if fill(st, 1) != Fill::Ok { return -1; } } st.cur += 1; yystate = 5; continue 'yyl; } _ => { panic!("internal lexer error") } } } } } } fn main() { let fname = "input"; let content = b"'qu\0tes' 'are' 'fine: \\'' "; // Prepare input file: a few times the size of the buffer, containing // strings with zeroes and escaped quotes. match File::create(fname) { Err(why) => panic!("cannot open {}: {}", fname, why), Ok(mut file) => match file.write_all(&content.repeat(BUFSIZE)) { Err(why) => panic!("cannot write to {}: {}", fname, why), Ok(_) => {} } }; let count = 3 * BUFSIZE; // number of quoted strings written to file // Reopen input file for reading. let file = match File::open(fname) { Err(why) => panic!("cannot read file {}: {}", fname, why), Ok(file) => file, }; // Initialize lexer state: all offsets are at the end of buffer. // This immediately triggers YYFILL, as the YYLESSTHAN condition is true. let lim = BUFSIZE - YYMAXFILL; let mut st = State { file: file, buf: [0; BUFSIZE], lim: lim, cur: lim, mar: lim, tok: lim, eof: false, }; // Run the lexer. assert_eq!(lex(&mut st), count as isize); // Cleanup: remove input file. match std::fs::remove_file(fname) { Err(why) => panic!("cannot remove {}: {}", fname, why), Ok(_) => {} } }