crispyconv/quirks.rs
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
/*
* SPDX-FileCopyrightText: 2022 Purism, SPC <https://puri.sm>
*
* SPDX-License-Identifier: MPL-2.0 OR LGPL-2.1-or-later
*/
use std::fmt;
macro_rules! ParseableEnum {
(
$(#[$a:meta])*
pub enum $name:ident {
$(
$(#[$attr:meta])*
$variant_name:ident,
)*
}
) => {
$(#[$a])*
pub enum $name {
$(
$(#[$attr])*
$variant_name,
)*
}
impl $name {
fn parse(value: &str) -> Result<Self, (ParsingError, &str)> {
$(
if value == stringify!($variant_name) {
return Ok(Self::$variant_name);
}
)*
return Err((ParsingError::Other("Invalid value"), value));
}
}
}
}
macro_rules! QuirksParser {
(
$(#[$a:meta])*
pub struct $name:ident {
$(
$(#[$attr:meta])*
pub $field_name:ident : $field_type:ty,
)*
}
) => {
$(#[$a])*
pub struct $name {
$(
$(#[$attr])*
pub $field_name: $field_type,
)*
}
impl $name {
pub fn parse(value: &str, mut base: Quirks)
-> Result<Self, (ParsingError, &str)>
{
fn parse_one<'a>(v: &'a str)
-> Result<[&'a str; 2], (ParsingError, &'a str)>
{
match &v.split("=").collect::<Vec<_>>()[..] {
[name, value] => Ok([name, value]),
_ => Err(
(ParsingError::Other("Not a name=value pair"), v)
),
}
}
for parsed in value.split(',').map(parse_one) {
let [name, value] = parsed?;
match name {
$(
stringify!($field_name) => {
base.$field_name = <$field_type>::parse(value)?;
},
)*
other => {
return Err((ParsingError::InvalidQuirk, other));
},
}
}
Ok(base)
}
pub fn names() -> &'static [&'static str] {
&[
$( stringify!($field_name), )*
]
}
}
}
}
#[derive(Debug)]
pub enum ParsingError {
InvalidQuirk,
Other(&'static str),
}
impl fmt::Display for ParsingError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> {
match self {
Self::InvalidQuirk => {
writeln!(f, "Invalid quirk. Allowed quirks:")?;
for name in Quirks::names() {
writeln!(f, " {}", name)?;
}
writeln!(f, "that's all.")?;
},
Self::Other(msg) => writeln!(f, "{}", msg)?,
};
Ok(())
}
}
ParseableEnum!{
#[derive(Clone, Debug)]
pub enum MultiByteHandling {
R16,
/// Lowest common denominator
R16AsG8R8,
}
}
QuirksParser! {
/// Stores hardware peculiarities.
#[derive(Clone, Debug)]
pub struct Quirks {
pub multi_byte_handling: MultiByteHandling,
}
}
pub const MOST_COMPATIBLE: Quirks = Quirks {
multi_byte_handling: MultiByteHandling::R16AsG8R8,
};
pub const MOST_PERFORMANT: Quirks = Quirks {
multi_byte_handling: MultiByteHandling::R16,
};