use alloc::string::String;
use core::{ffi::c_char, fmt};
use axio::Write;
use crate::ctypes;
pub trait WriteByte: fmt::Write {
fn write_u8(&mut self, byte: u8) -> fmt::Result;
}
struct StringWriter(pub *mut u8, pub usize);
impl Write for StringWriter {
fn write(&mut self, buf: &[u8]) -> axerrno::AxResult<usize> {
if self.1 > 1 {
let copy_size = buf.len().min(self.1 - 1);
unsafe {
core::ptr::copy_nonoverlapping(buf.as_ptr(), self.0, copy_size);
self.1 -= copy_size;
self.0 = self.0.add(copy_size);
*self.0 = 0;
}
}
Ok(buf.len())
}
fn flush(&mut self) -> axerrno::AxResult {
Ok(())
}
}
impl fmt::Write for StringWriter {
fn write_str(&mut self, s: &str) -> fmt::Result {
self.write(s.as_bytes()).unwrap();
Ok(())
}
}
impl WriteByte for StringWriter {
fn write_u8(&mut self, byte: u8) -> fmt::Result {
self.write(&[byte]).unwrap();
Ok(())
}
}
struct CountingWriter<T> {
pub inner: T,
pub written: usize,
}
impl<T> CountingWriter<T> {
pub fn new(writer: T) -> Self {
Self {
inner: writer,
written: 0,
}
}
}
impl<T: fmt::Write> fmt::Write for CountingWriter<T> {
fn write_str(&mut self, s: &str) -> fmt::Result {
self.written += s.len();
self.inner.write_str(s)
}
}
impl<T: WriteByte> WriteByte for CountingWriter<T> {
fn write_u8(&mut self, byte: u8) -> fmt::Result {
self.written += 1;
self.inner.write_u8(byte)
}
}
impl<T: Write> Write for CountingWriter<T> {
fn write(&mut self, buf: &[u8]) -> axerrno::AxResult<usize> {
let res = self.inner.write(buf);
if let Ok(written) = res {
self.written += written;
}
res
}
fn write_all(&mut self, buf: &[u8]) -> axerrno::AxResult {
match self.inner.write_all(buf) {
Ok(()) => (),
Err(err) => return Err(err),
}
self.written += buf.len();
Ok(())
}
fn flush(&mut self) -> axerrno::AxResult {
self.inner.flush()
}
}
unsafe fn strftime_inner<W: WriteByte>(
w: W,
format: *const c_char,
t: *const ctypes::tm,
) -> ctypes::size_t {
#[allow(clippy::unnecessary_cast)]
pub unsafe fn inner_strftime<W: WriteByte>(
w: &mut W,
mut format: *const c_char,
t: *const ctypes::tm,
) -> bool {
macro_rules! w {
(byte $b:expr) => {{
if w.write_u8($b).is_err() {
return false;
}
}};
(char $chr:expr) => {{
if w.write_char($chr).is_err() {
return false;
}
}};
(recurse $fmt:expr) => {{
let mut fmt = String::with_capacity($fmt.len() + 1);
fmt.push_str($fmt);
fmt.push('\0');
if !inner_strftime(w, fmt.as_ptr() as *mut c_char, t) {
return false;
}
}};
($str:expr) => {{
if w.write_str($str).is_err() {
return false;
}
}};
($fmt:expr, $($args:expr),+) => {{
if write!(w, $fmt, $($args),+).is_err() {
return false;
}
}};
}
const WDAYS: [&str; 7] = [
"Sunday",
"Monday",
"Tuesday",
"Wednesday",
"Thursday",
"Friday",
"Saturday",
];
const MONTHS: [&str; 12] = [
"January",
"Febuary",
"March",
"April",
"May",
"June",
"July",
"August",
"September",
"October",
"November",
"December",
];
while *format != 0 {
if *format as u8 != b'%' {
w!(byte * format as u8);
format = format.offset(1);
continue;
}
format = format.offset(1);
if *format as u8 == b'E' || *format as u8 == b'O' {
format = format.offset(1);
}
match *format as u8 {
b'%' => w!(byte b'%'),
b'n' => w!(byte b'\n'),
b't' => w!(byte b'\t'),
b'a' => w!(&WDAYS[(*t).tm_wday as usize][..3]),
b'A' => w!(WDAYS[(*t).tm_wday as usize]),
b'b' | b'h' => w!(&MONTHS[(*t).tm_mon as usize][..3]),
b'B' => w!(MONTHS[(*t).tm_mon as usize]),
b'C' => {
let mut year = (*t).tm_year / 100;
if (*t).tm_year % 100 != 0 {
year += 1;
}
w!("{:02}", year + 19);
}
b'd' => w!("{:02}", (*t).tm_mday),
b'D' => w!(recurse "%m/%d/%y"),
b'e' => w!("{:2}", (*t).tm_mday),
b'F' => w!(recurse "%Y-%m-%d"),
b'H' => w!("{:02}", (*t).tm_hour),
b'I' => w!("{:02}", ((*t).tm_hour + 12 - 1) % 12 + 1),
b'j' => w!("{:03}", (*t).tm_yday),
b'k' => w!("{:2}", (*t).tm_hour),
b'l' => w!("{:2}", ((*t).tm_hour + 12 - 1) % 12 + 1),
b'm' => w!("{:02}", (*t).tm_mon + 1),
b'M' => w!("{:02}", (*t).tm_min),
b'p' => w!(if (*t).tm_hour < 12 { "AM" } else { "PM" }),
b'P' => w!(if (*t).tm_hour < 12 { "am" } else { "pm" }),
b'r' => w!(recurse "%I:%M:%S %p"),
b'R' => w!(recurse "%H:%M"),
b's' => w!("{}", super::mktime(t as *mut ctypes::tm)),
b'S' => w!("{:02}", (*t).tm_sec),
b'T' => w!(recurse "%H:%M:%S"),
b'u' => w!("{}", ((*t).tm_wday + 7 - 1) % 7 + 1),
b'U' => w!("{}", ((*t).tm_yday + 7 - (*t).tm_wday) / 7),
b'w' => w!("{}", (*t).tm_wday),
b'W' => w!("{}", ((*t).tm_yday + 7 - ((*t).tm_wday + 6) % 7) / 7),
b'y' => w!("{:02}", (*t).tm_year % 100),
b'Y' => w!("{}", (*t).tm_year + 1900),
b'z' => w!("+0000"), b'Z' => w!("UTC"), b'+' => w!(recurse "%a %b %d %T %Z %Y"),
_ => return false,
}
format = format.offset(1);
}
true
}
let mut w: CountingWriter<W> = CountingWriter::new(w);
if !inner_strftime(&mut w, format, t) {
return 0;
}
w.written
}
#[unsafe(no_mangle)]
#[allow(clippy::unnecessary_cast)] pub unsafe extern "C" fn strftime(
buf: *mut c_char,
size: ctypes::size_t,
format: *const c_char,
timeptr: *const ctypes::tm,
) -> ctypes::size_t {
let ret = strftime_inner(StringWriter(buf as *mut u8, size), format, timeptr);
if ret < size { ret } else { 0 }
}