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
|
// SPDX-License-Identifier: AGPL-3.0-or-later
//! Executor engine implementation for Brainfuck interpreter.
//!
//! This predominantly allows implementation of a [`u16`] executor.
#[cfg(feature = "bigint-engine")]
use byteorder::{NativeEndian, ReadBytesExt};
use num_traits::{One, Unsigned, WrappingAdd, WrappingSub, Zero};
use thiserror::Error;
use crate::executor;
/// Error type for engine errors
#[derive(Debug, Error)]
pub enum Error {
/// Engine ran into an io Error
#[error(transparent)]
Io(#[from] std::io::Error),
/// Utf16 error from widestring
#[error("could not convert byte to Utf16Str")]
Utf16,
/// Utf32 error from widestring
#[error("could not convert byte to Utf32Str")]
Utf32,
}
/// Generic engine implementation for the Brainfuck interpreter.
pub trait Engine {
/// Inner type of the Tape.
type TapeInner: Clone + Copy + Unsigned + WrappingAdd + WrappingSub + One + Zero;
/// Read one byte from stdin.
///
/// # Errors
///
/// This function will return an error if it is unable to read from stdin,
/// or if it indexes out of bounds.
fn read_byte() -> Result<Self::TapeInner, Error>;
/// Write the provided byte to stdout.
///
/// # Errors
///
/// This function will return an error if it is unable to write a byte to
/// stdout.
fn write_byte(byte: Self::TapeInner) -> Result<(), Error>;
}
impl Engine for executor::U8 {
type TapeInner = u8;
fn read_byte() -> Result<u8, Error> {
Ok(std::io::stdin().read_u8()?)
}
fn write_byte(byte: u8) -> Result<(), Error> {
print!("{}", char::from(byte));
Ok(())
}
}
#[cfg(feature = "engine-u16")]
impl Engine for executor::U16 {
type TapeInner = u16;
fn read_byte() -> Result<u16, Error> {
Ok(std::io::stdin().read_u16::<NativeEndian>()?)
}
fn write_byte(byte: u16) -> Result<(), Error> {
print!(
"{}",
widestring::Utf16Str::from_slice(&[byte]).map_err(|_err| Error::Utf16)?
);
Ok(())
}
}
#[cfg(feature = "engine-u32")]
impl Engine for executor::U32 {
type TapeInner = u32;
fn read_byte() -> Result<u32, Error> {
Ok(std::io::stdin().read_u32::<NativeEndian>()?)
}
fn write_byte(byte: u32) -> Result<(), Error> {
print!(
"{}",
widestring::Utf32Str::from_slice(&[byte]).map_err(|_err| Error::Utf32)?
);
Ok(())
}
}
|