Skip to content

Commit

Permalink
Merge f7fe0b9 into 65f9105
Browse files Browse the repository at this point in the history
  • Loading branch information
HalidOdat authored Apr 20, 2023
2 parents 65f9105 + f7fe0b9 commit c4093a1
Showing 1 changed file with 69 additions and 19 deletions.
88 changes: 69 additions & 19 deletions boa_engine/src/property/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@ mod attribute;

use crate::{js_string, JsString, JsSymbol, JsValue};
use boa_gc::{Finalize, Trace};
use std::fmt;
use std::{fmt, iter::FusedIterator};

pub use attribute::Attribute;

Expand Down Expand Up @@ -569,52 +569,102 @@ pub enum PropertyKey {
Index(u32),
}

/// Utility function for parsing [`PropertyKey`].
fn parse_u32_index<I, T>(mut input: I) -> Option<u32>
where
I: Iterator<Item = T> + ExactSizeIterator + FusedIterator,
T: Into<u16>,
{
// min: 0 --> 1 char
// max: 4_294_967_296 --> 10 chars
//
// Max char range: [1, 10] inclusive.
const MAX_CHAR_COUNT: usize = 10;

const CHAR_ZERO: u16 = b'0' as u16;
const CHAR_NINE: u16 = b'9' as u16;

// Eliminate any string if it's greater than the max char count.
let len = input.len();
if len > MAX_CHAR_COUNT {
return None;
}

// Helper function, for converting character to digit [0, 9].
let to_digit = |c: u16| -> Option<u32> {
if matches!(c, CHAR_ZERO..=CHAR_NINE) {
Some(u32::from(c - CHAR_ZERO))
} else {
None
}
};

let byte = input.next()?.into();
if byte == CHAR_ZERO {
if len == 1 {
return Some(0);
}

// String "012345" is not a valid index.
return None;
}

let mut result = to_digit(byte)?;

// If the len is equal to max chars, then we need to do checked opterations,
// in case of overflows. If less use unchecked versions.
if len == MAX_CHAR_COUNT {
for c in input {
result = result.checked_mul(10)?.checked_add(to_digit(c.into())?)?;
}
} else {
for c in input {
result = result * 10 + to_digit(c.into())?;
}
}

Some(result)
}

impl From<&[u16]> for PropertyKey {
#[inline]
fn from(string: &[u16]) -> Self {
debug_assert!(String::from_utf16(string)
.expect("should be ascii string")
.parse::<u32>()
.is_err());
debug_assert!(parse_u32_index(
String::from_utf16(string)
.expect("should be ascii string")
.bytes()
)
.is_none());
Self::String(string.into())
}
}

impl From<JsString> for PropertyKey {
#[inline]
fn from(string: JsString) -> Self {
string
.to_std_string()
.ok()
.and_then(|s| s.parse().ok())
.map_or(Self::String(string), Self::Index)
parse_u32_index(string.as_slice().iter().copied()).map_or(Self::String(string), Self::Index)
}
}

impl From<&str> for PropertyKey {
#[inline]
fn from(string: &str) -> Self {
string
.parse()
.map_or_else(|_| Self::String(string.into()), Self::Index)
parse_u32_index(string.bytes()).map_or_else(|| Self::String(string.into()), Self::Index)
}
}

impl From<String> for PropertyKey {
#[inline]
fn from(string: String) -> Self {
string
.parse()
.map_or_else(|_| Self::String(string.into()), Self::Index)
parse_u32_index(string.bytes()).map_or_else(|| Self::String(string.into()), Self::Index)
}
}

impl From<Box<str>> for PropertyKey {
#[inline]
fn from(string: Box<str>) -> Self {
string
.parse()
.map_or_else(|_| Self::String(string.as_ref().into()), Self::Index)
parse_u32_index(string.bytes())
.map_or_else(|| Self::String(string.as_ref().into()), Self::Index)
}
}

Expand Down

0 comments on commit c4093a1

Please sign in to comment.