|  | @@ -1,7 +1,8 @@
 | 
	
		
			
				|  |  |  use alloc::string::String;
 | 
	
		
			
				|  |  |  use alloc::string::ToString;
 | 
	
		
			
				|  |  | +use alloc::collections::BTreeMap;
 | 
	
		
			
				|  |  |  use alloc::vec::Vec;
 | 
	
		
			
				|  |  | -use core::ffi::VaList as va_list;
 | 
	
		
			
				|  |  | +use core::ffi::VaList;
 | 
	
		
			
				|  |  |  use core::ops::Range;
 | 
	
		
			
				|  |  |  use core::{fmt, slice};
 | 
	
		
			
				|  |  |  use io::{self, Write};
 | 
	
	
		
			
				|  | @@ -9,7 +10,15 @@ use io::{self, Write};
 | 
	
		
			
				|  |  |  use platform;
 | 
	
		
			
				|  |  |  use platform::types::*;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -#[derive(PartialEq, Eq)]
 | 
	
		
			
				|  |  | +//  ____        _ _                 _       _
 | 
	
		
			
				|  |  | +// | __ )  ___ (_) | ___ _ __ _ __ | | __ _| |_ ___ _
 | 
	
		
			
				|  |  | +// |  _ \ / _ \| | |/ _ \ '__| '_ \| |/ _` | __/ _ (_)
 | 
	
		
			
				|  |  | +// | |_) | (_) | | |  __/ |  | |_) | | (_| | ||  __/_
 | 
	
		
			
				|  |  | +// |____/ \___/|_|_|\___|_|  | .__/|_|\__,_|\__\___(_)
 | 
	
		
			
				|  |  | +//                           |_|
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +#[derive(Clone, Copy, PartialEq, Eq)]
 | 
	
		
			
				|  |  |  enum IntKind {
 | 
	
		
			
				|  |  |      Byte,
 | 
	
		
			
				|  |  |      Short,
 | 
	
	
		
			
				|  | @@ -20,6 +29,7 @@ enum IntKind {
 | 
	
		
			
				|  |  |      PtrDiff,
 | 
	
		
			
				|  |  |      Size,
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  | +#[derive(Clone, Copy, PartialEq, Eq)]
 | 
	
		
			
				|  |  |  enum FmtKind {
 | 
	
		
			
				|  |  |      Percent,
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -42,144 +52,113 @@ enum Number {
 | 
	
		
			
				|  |  |      Next
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  impl Number {
 | 
	
		
			
				|  |  | -    unsafe fn resolve(&self, ap: &mut BufferedVaList) -> usize {
 | 
	
		
			
				|  |  | -        match *self {
 | 
	
		
			
				|  |  | -            Number::Static(num) => num,
 | 
	
		
			
				|  |  | -            Number::Index(i) => ap.index(ArgKind::Int, i).int as usize,
 | 
	
		
			
				|  |  | -            Number::Next => ap.next(ArgKind::Int).int as usize,
 | 
	
		
			
				|  |  | +    unsafe fn resolve(&self, varargs: &mut VaListCache, ap: &mut VaList) -> usize {
 | 
	
		
			
				|  |  | +        let arg = match *self {
 | 
	
		
			
				|  |  | +            Number::Static(num) => return num,
 | 
	
		
			
				|  |  | +            Number::Index(i) => varargs.get(i-1, ap, None),
 | 
	
		
			
				|  |  | +            Number::Next => {
 | 
	
		
			
				|  |  | +                let i = varargs.i;
 | 
	
		
			
				|  |  | +                varargs.i += 1;
 | 
	
		
			
				|  |  | +                varargs.get(i, ap, None)
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +        };
 | 
	
		
			
				|  |  | +        match arg {
 | 
	
		
			
				|  |  | +            VaArg::c_char(i) => i as usize,
 | 
	
		
			
				|  |  | +            VaArg::c_double(i) => i as usize,
 | 
	
		
			
				|  |  | +            VaArg::c_int(i) => i as usize,
 | 
	
		
			
				|  |  | +            VaArg::c_long(i) => i as usize,
 | 
	
		
			
				|  |  | +            VaArg::c_longlong(i) => i as usize,
 | 
	
		
			
				|  |  | +            VaArg::c_short(i) => i as usize,
 | 
	
		
			
				|  |  | +            VaArg::intmax_t(i) => i as usize,
 | 
	
		
			
				|  |  | +            VaArg::pointer(i) => i as usize,
 | 
	
		
			
				|  |  | +            VaArg::ptrdiff_t(i) => i as usize,
 | 
	
		
			
				|  |  | +            VaArg::ssize_t(i) => i as usize
 | 
	
		
			
				|  |  |          }
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  | -enum ArgKind {
 | 
	
		
			
				|  |  | -    Byte,
 | 
	
		
			
				|  |  | -    Short,
 | 
	
		
			
				|  |  | -    Int,
 | 
	
		
			
				|  |  | -    Long,
 | 
	
		
			
				|  |  | -    LongLong,
 | 
	
		
			
				|  |  | -    PtrDiff,
 | 
	
		
			
				|  |  | -    Size,
 | 
	
		
			
				|  |  | -    IntMax,
 | 
	
		
			
				|  |  | -    Double,
 | 
	
		
			
				|  |  | -    CharPtr,
 | 
	
		
			
				|  |  | -    VoidPtr,
 | 
	
		
			
				|  |  | -    IntPtr,
 | 
	
		
			
				|  |  | -    ArgDefault,
 | 
	
		
			
				|  |  | -}
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  |  #[derive(Clone, Copy)]
 | 
	
		
			
				|  |  | -union VaArg {
 | 
	
		
			
				|  |  | -    byte: c_char,
 | 
	
		
			
				|  |  | -    short: c_short,
 | 
	
		
			
				|  |  | -    int: c_int,
 | 
	
		
			
				|  |  | -    long: c_long,
 | 
	
		
			
				|  |  | -    longlong: c_longlong,
 | 
	
		
			
				|  |  | -    ptrdiff: ptrdiff_t,
 | 
	
		
			
				|  |  | -    size: ssize_t,
 | 
	
		
			
				|  |  | -    intmax: intmax_t,
 | 
	
		
			
				|  |  | -    double: c_double,
 | 
	
		
			
				|  |  | -    char_ptr: *const c_char,
 | 
	
		
			
				|  |  | -    void_ptr: *const c_void,
 | 
	
		
			
				|  |  | -    int_ptr: *mut c_int,
 | 
	
		
			
				|  |  | -    arg_default: usize,
 | 
	
		
			
				|  |  | -}
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -struct BufferedVaList<'a> {
 | 
	
		
			
				|  |  | -    list: va_list<'a>,
 | 
	
		
			
				|  |  | -    buf: Vec<VaArg>,
 | 
	
		
			
				|  |  | -    i: usize,
 | 
	
		
			
				|  |  | +enum VaArg {
 | 
	
		
			
				|  |  | +    c_char(c_char),
 | 
	
		
			
				|  |  | +    c_double(c_double),
 | 
	
		
			
				|  |  | +    c_int(c_int),
 | 
	
		
			
				|  |  | +    c_long(c_long),
 | 
	
		
			
				|  |  | +    c_longlong(c_longlong),
 | 
	
		
			
				|  |  | +    c_short(c_short),
 | 
	
		
			
				|  |  | +    intmax_t(intmax_t),
 | 
	
		
			
				|  |  | +    pointer(*const c_void),
 | 
	
		
			
				|  |  | +    ptrdiff_t(ptrdiff_t),
 | 
	
		
			
				|  |  | +    ssize_t(ssize_t)
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -impl<'a> BufferedVaList<'a> {
 | 
	
		
			
				|  |  | -    fn new(list: va_list<'a>) -> Self {
 | 
	
		
			
				|  |  | -        Self {
 | 
	
		
			
				|  |  | -            list,
 | 
	
		
			
				|  |  | -            buf: Vec::new(),
 | 
	
		
			
				|  |  | -            i: 0,
 | 
	
		
			
				|  |  | +impl VaArg {
 | 
	
		
			
				|  |  | +    unsafe fn arg_from(arg: &PrintfArg, ap: &mut VaList) -> VaArg {
 | 
	
		
			
				|  |  | +        // Per the C standard using va_arg with a type with a size
 | 
	
		
			
				|  |  | +        // less than that of an int for integers and double for floats
 | 
	
		
			
				|  |  | +        // is invalid. As a result any arguments smaller than an int or
 | 
	
		
			
				|  |  | +        // double passed to a function will be promoted to the smallest
 | 
	
		
			
				|  |  | +        // possible size. The VaList::arg function will handle this
 | 
	
		
			
				|  |  | +        // automagically.
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        match (arg.fmtkind, arg.intkind) {
 | 
	
		
			
				|  |  | +            (FmtKind::Percent, _) => panic!("Can't call arg_from on %"),
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            (FmtKind::Char, _) |
 | 
	
		
			
				|  |  | +            (FmtKind::Unsigned, IntKind::Byte) |
 | 
	
		
			
				|  |  | +            (FmtKind::Signed, IntKind::Byte) => VaArg::c_char(ap.arg::<c_char>()),
 | 
	
		
			
				|  |  | +            (FmtKind::Unsigned, IntKind::Short) |
 | 
	
		
			
				|  |  | +            (FmtKind::Signed, IntKind::Short) => VaArg::c_short(ap.arg::<c_short>()),
 | 
	
		
			
				|  |  | +            (FmtKind::Unsigned, IntKind::Int) |
 | 
	
		
			
				|  |  | +            (FmtKind::Signed, IntKind::Int) => VaArg::c_int(ap.arg::<c_int>()),
 | 
	
		
			
				|  |  | +            (FmtKind::Unsigned, IntKind::Long) |
 | 
	
		
			
				|  |  | +            (FmtKind::Signed, IntKind::Long) => VaArg::c_long(ap.arg::<c_long>()),
 | 
	
		
			
				|  |  | +            (FmtKind::Unsigned, IntKind::LongLong) |
 | 
	
		
			
				|  |  | +            (FmtKind::Signed, IntKind::LongLong) => VaArg::c_longlong(ap.arg::<c_longlong>()),
 | 
	
		
			
				|  |  | +            (FmtKind::Unsigned, IntKind::IntMax) |
 | 
	
		
			
				|  |  | +            (FmtKind::Signed, IntKind::IntMax) => VaArg::intmax_t(ap.arg::<intmax_t>()),
 | 
	
		
			
				|  |  | +            (FmtKind::Unsigned, IntKind::PtrDiff) |
 | 
	
		
			
				|  |  | +            (FmtKind::Signed, IntKind::PtrDiff) => VaArg::ptrdiff_t(ap.arg::<ptrdiff_t>()),
 | 
	
		
			
				|  |  | +            (FmtKind::Unsigned, IntKind::Size) |
 | 
	
		
			
				|  |  | +            (FmtKind::Signed, IntKind::Size) => VaArg::ssize_t(ap.arg::<ssize_t>()),
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            (FmtKind::AnyNotation, _) | (FmtKind::Decimal, _) | (FmtKind::Scientific, _)
 | 
	
		
			
				|  |  | +                => VaArg::c_double(ap.arg::<c_double>()),
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            (FmtKind::GetWritten, _) | (FmtKind::Pointer, _) | (FmtKind::String, _)
 | 
	
		
			
				|  |  | +                => VaArg::pointer(ap.arg::<*const c_void>()),
 | 
	
		
			
				|  |  |          }
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -    unsafe fn get_arg(&mut self, ty: ArgKind) -> VaArg {
 | 
	
		
			
				|  |  | -        match ty {
 | 
	
		
			
				|  |  | -            ArgKind::Byte => VaArg {
 | 
	
		
			
				|  |  | -                byte: self.list.arg::<c_char>(),
 | 
	
		
			
				|  |  | -            },
 | 
	
		
			
				|  |  | -            ArgKind::Short => VaArg {
 | 
	
		
			
				|  |  | -                short: self.list.arg::<c_short>(),
 | 
	
		
			
				|  |  | -            },
 | 
	
		
			
				|  |  | -            ArgKind::Int => VaArg {
 | 
	
		
			
				|  |  | -                int: self.list.arg::<c_int>(),
 | 
	
		
			
				|  |  | -            },
 | 
	
		
			
				|  |  | -            ArgKind::Long => VaArg {
 | 
	
		
			
				|  |  | -                long: self.list.arg::<c_long>(),
 | 
	
		
			
				|  |  | -            },
 | 
	
		
			
				|  |  | -            ArgKind::LongLong => VaArg {
 | 
	
		
			
				|  |  | -                longlong: self.list.arg::<c_longlong>(),
 | 
	
		
			
				|  |  | -            },
 | 
	
		
			
				|  |  | -            ArgKind::PtrDiff => VaArg {
 | 
	
		
			
				|  |  | -                ptrdiff: self.list.arg::<ptrdiff_t>(),
 | 
	
		
			
				|  |  | -            },
 | 
	
		
			
				|  |  | -            ArgKind::Size => VaArg {
 | 
	
		
			
				|  |  | -                size: self.list.arg::<ssize_t>(),
 | 
	
		
			
				|  |  | -            },
 | 
	
		
			
				|  |  | -            ArgKind::IntMax => VaArg {
 | 
	
		
			
				|  |  | -                intmax: self.list.arg::<intmax_t>(),
 | 
	
		
			
				|  |  | -            },
 | 
	
		
			
				|  |  | -            ArgKind::Double => VaArg {
 | 
	
		
			
				|  |  | -                double: self.list.arg::<c_double>(),
 | 
	
		
			
				|  |  | -            },
 | 
	
		
			
				|  |  | -            ArgKind::CharPtr => VaArg {
 | 
	
		
			
				|  |  | -                char_ptr: self.list.arg::<*const c_char>(),
 | 
	
		
			
				|  |  | -            },
 | 
	
		
			
				|  |  | -            ArgKind::VoidPtr => VaArg {
 | 
	
		
			
				|  |  | -                void_ptr: self.list.arg::<*const c_void>(),
 | 
	
		
			
				|  |  | -            },
 | 
	
		
			
				|  |  | -            ArgKind::IntPtr => VaArg {
 | 
	
		
			
				|  |  | -                int_ptr: self.list.arg::<*mut c_int>(),
 | 
	
		
			
				|  |  | -            },
 | 
	
		
			
				|  |  | -            ArgKind::ArgDefault => VaArg {
 | 
	
		
			
				|  |  | -                arg_default: self.list.arg::<usize>(),
 | 
	
		
			
				|  |  | -            },
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +#[derive(Default)]
 | 
	
		
			
				|  |  | +struct VaListCache {
 | 
	
		
			
				|  |  | +    args: Vec<VaArg>,
 | 
	
		
			
				|  |  | +    i: usize
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +impl VaListCache {
 | 
	
		
			
				|  |  | +    unsafe fn get(&mut self, i: usize, ap: &mut VaList, arg: Option<&PrintfArg>) -> VaArg {
 | 
	
		
			
				|  |  | +        if let Some(&arg) = self.args.get(i) {
 | 
	
		
			
				|  |  | +            return arg;
 | 
	
		
			
				|  |  |          }
 | 
	
		
			
				|  |  | -    }
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -    unsafe fn get(&mut self, ty: ArgKind, i: Option<usize>) -> VaArg {
 | 
	
		
			
				|  |  | -        match i {
 | 
	
		
			
				|  |  | -            None => self.next(ty),
 | 
	
		
			
				|  |  | -            Some(i) => self.index(ty, i),
 | 
	
		
			
				|  |  | +        while self.args.len() < i {
 | 
	
		
			
				|  |  | +            // We can't POSSIBLY know the type if we reach this
 | 
	
		
			
				|  |  | +            // point. Reaching here means there are unused gaps in the
 | 
	
		
			
				|  |  | +            // arguments. Ultimately we'll have to settle down with
 | 
	
		
			
				|  |  | +            // defaulting to c_int.
 | 
	
		
			
				|  |  | +            self.args.push(VaArg::c_int(ap.arg::<c_int>()))
 | 
	
		
			
				|  |  |          }
 | 
	
		
			
				|  |  | +        self.args.push(match arg {
 | 
	
		
			
				|  |  | +            Some(arg) => VaArg::arg_from(arg, ap),
 | 
	
		
			
				|  |  | +            None => VaArg::c_int(ap.arg::<c_int>())
 | 
	
		
			
				|  |  | +        });
 | 
	
		
			
				|  |  | +        self.args[i]
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -    unsafe fn next(&mut self, ty: ArgKind) -> VaArg {
 | 
	
		
			
				|  |  | -        if self.i >= self.buf.len() {
 | 
	
		
			
				|  |  | -            let arg = self.get_arg(ty);
 | 
	
		
			
				|  |  | -            self.buf.push(arg);
 | 
	
		
			
				|  |  | -        }
 | 
	
		
			
				|  |  | -        let arg = self.buf[self.i];
 | 
	
		
			
				|  |  | -        self.i += 1;
 | 
	
		
			
				|  |  | -        arg
 | 
	
		
			
				|  |  | -    }
 | 
	
		
			
				|  |  | +//  ___                 _                           _        _   _
 | 
	
		
			
				|  |  | +// |_ _|_ __ ___  _ __ | | ___ _ __ ___   ___ _ __ | |_ __ _| |_(_) ___  _ __  _
 | 
	
		
			
				|  |  | +//  | || '_ ` _ \| '_ \| |/ _ \ '_ ` _ \ / _ \ '_ \| __/ _` | __| |/ _ \| '_ \(_)
 | 
	
		
			
				|  |  | +//  | || | | | | | |_) | |  __/ | | | | |  __/ | | | || (_| | |_| | (_) | | | |_
 | 
	
		
			
				|  |  | +// |___|_| |_| |_| .__/|_|\___|_| |_| |_|\___|_| |_|\__\__,_|\__|_|\___/|_| |_(_)
 | 
	
		
			
				|  |  | +//               |_|
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -    unsafe fn index(&mut self, ty: ArgKind, i: usize) -> VaArg {
 | 
	
		
			
				|  |  | -        if self.i >= self.buf.len() {
 | 
	
		
			
				|  |  | -            while self.buf.len() < (i - 1) {
 | 
	
		
			
				|  |  | -                // Getting a usize here most definitely isn't sane, however,
 | 
	
		
			
				|  |  | -                // there's no way to know the type!
 | 
	
		
			
				|  |  | -                // Just take this for example:
 | 
	
		
			
				|  |  | -                //
 | 
	
		
			
				|  |  | -                // printf("%*4$d\n", "hi", 0, "hello", 10);
 | 
	
		
			
				|  |  | -                //
 | 
	
		
			
				|  |  | -                // This chooses the width 10. How does it know the type of 0 and "hello"?
 | 
	
		
			
				|  |  | -                // It clearly can't.
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -                let arg = self.get_arg(ArgKind::ArgDefault);
 | 
	
		
			
				|  |  | -                self.buf.push(arg);
 | 
	
		
			
				|  |  | -            }
 | 
	
		
			
				|  |  | -            let arg = self.get_arg(ty);
 | 
	
		
			
				|  |  | -            self.buf.push(arg);
 | 
	
		
			
				|  |  | -        }
 | 
	
		
			
				|  |  | -        self.buf[i - 1]
 | 
	
		
			
				|  |  | -    }
 | 
	
		
			
				|  |  | -}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  unsafe fn pop_int_raw(format: &mut *const u8) -> Option<usize> {
 | 
	
		
			
				|  |  |      let mut int = None;
 | 
	
	
		
			
				|  | @@ -342,6 +321,7 @@ fn fmt_float_normal<W: Write>(
 | 
	
		
			
				|  |  |  struct PrintfIter {
 | 
	
		
			
				|  |  |      format: *const u8
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  | +#[derive(Clone, Copy)]
 | 
	
		
			
				|  |  |  struct PrintfArg {
 | 
	
		
			
				|  |  |      index: Option<usize>,
 | 
	
		
			
				|  |  |      alternate: bool,
 | 
	
	
		
			
				|  | @@ -486,14 +466,40 @@ impl Iterator for PrintfIter {
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -unsafe fn inner_printf<W: Write>(w: W, format: *const c_char, ap: va_list) -> io::Result<c_int> {
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +unsafe fn inner_printf<W: Write>(w: W, format: *const c_char, mut ap: VaList) -> io::Result<c_int> {
 | 
	
		
			
				|  |  |      let w = &mut platform::CountingWriter::new(w);
 | 
	
		
			
				|  |  | -    let mut ap = BufferedVaList::new(ap);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      let iterator = PrintfIter {
 | 
	
		
			
				|  |  |          format: format as *const u8
 | 
	
		
			
				|  |  |      };
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +    // Pre-fetch vararg types
 | 
	
		
			
				|  |  | +    let mut varargs = VaListCache::default();
 | 
	
		
			
				|  |  | +    let mut positional = BTreeMap::new();
 | 
	
		
			
				|  |  | +    // ^ NOTE: This depends on the sorted order, do not change to HashMap or whatever
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    for section in iterator {
 | 
	
		
			
				|  |  | +        let arg = match section {
 | 
	
		
			
				|  |  | +            Ok(PrintfFmt::Plain(text)) => continue,
 | 
	
		
			
				|  |  | +            Ok(PrintfFmt::Arg(arg)) => arg,
 | 
	
		
			
				|  |  | +            Err(()) => return Ok(-1)
 | 
	
		
			
				|  |  | +        };
 | 
	
		
			
				|  |  | +        if arg.fmtkind == FmtKind::Percent {
 | 
	
		
			
				|  |  | +            continue;
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +        if let Some(i) = arg.index {
 | 
	
		
			
				|  |  | +            positional.insert(i-1, arg);
 | 
	
		
			
				|  |  | +        } else {
 | 
	
		
			
				|  |  | +            varargs.args.push(VaArg::arg_from(&arg, &mut ap));
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +    // Make sure, in order, the positional arguments exist with the specified type
 | 
	
		
			
				|  |  | +    for (i, arg) in positional {
 | 
	
		
			
				|  |  | +        varargs.get(i, &mut ap, Some(&arg));
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    // Main loop
 | 
	
		
			
				|  |  |      for section in iterator {
 | 
	
		
			
				|  |  |          let arg = match section {
 | 
	
		
			
				|  |  |              Ok(PrintfFmt::Plain(text)) => {
 | 
	
	
		
			
				|  | @@ -503,40 +509,43 @@ unsafe fn inner_printf<W: Write>(w: W, format: *const c_char, ap: va_list) -> io
 | 
	
		
			
				|  |  |              Ok(PrintfFmt::Arg(arg)) => arg,
 | 
	
		
			
				|  |  |              Err(()) => return Ok(-1)
 | 
	
		
			
				|  |  |          };
 | 
	
		
			
				|  |  | -        let index = arg.index;
 | 
	
		
			
				|  |  |          let alternate = arg.alternate;
 | 
	
		
			
				|  |  |          let zero = arg.zero;
 | 
	
		
			
				|  |  |          let left = arg.left;
 | 
	
		
			
				|  |  |          let sign_reserve = arg.sign_reserve;
 | 
	
		
			
				|  |  |          let sign_always = arg.sign_always;
 | 
	
		
			
				|  |  | -        let min_width = arg.min_width.resolve(&mut ap);
 | 
	
		
			
				|  |  | -        let precision = arg.precision.map(|n| n.resolve(&mut ap));
 | 
	
		
			
				|  |  | -        let pad_space = arg.pad_space.resolve(&mut ap);
 | 
	
		
			
				|  |  | -        let pad_zero = arg.pad_zero.resolve(&mut ap);
 | 
	
		
			
				|  |  | +        let min_width = arg.min_width.resolve(&mut varargs, &mut ap);
 | 
	
		
			
				|  |  | +        let precision = arg.precision.map(|n| n.resolve(&mut varargs, &mut ap));
 | 
	
		
			
				|  |  | +        let pad_space = arg.pad_space.resolve(&mut varargs, &mut ap);
 | 
	
		
			
				|  |  | +        let pad_zero = arg.pad_zero.resolve(&mut varargs, &mut ap);
 | 
	
		
			
				|  |  |          let intkind = arg.intkind;
 | 
	
		
			
				|  |  |          let fmt = arg.fmt;
 | 
	
		
			
				|  |  |          let fmtkind = arg.fmtkind;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -        // Finally, type:
 | 
	
		
			
				|  |  | +        let index = arg.index
 | 
	
		
			
				|  |  | +            .map(|i| i-1)
 | 
	
		
			
				|  |  | +            .unwrap_or_else(|| if fmtkind == FmtKind::Percent {
 | 
	
		
			
				|  |  | +                0
 | 
	
		
			
				|  |  | +            } else {
 | 
	
		
			
				|  |  | +                let i = varargs.i;
 | 
	
		
			
				|  |  | +                varargs.i += 1;
 | 
	
		
			
				|  |  | +                i
 | 
	
		
			
				|  |  | +            });
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |          match fmtkind {
 | 
	
		
			
				|  |  |              FmtKind::Percent => w.write_all(&[b'%'])?,
 | 
	
		
			
				|  |  |              FmtKind::Signed => {
 | 
	
		
			
				|  |  | -                let string = match intkind {
 | 
	
		
			
				|  |  | -                    // Per the C standard using va_arg with a type with a size
 | 
	
		
			
				|  |  | -                    // less than that of an int for integers and double for floats
 | 
	
		
			
				|  |  | -                    // is invalid. As a result any arguments smaller than an int or
 | 
	
		
			
				|  |  | -                    // double passed to a function will be promoted to the smallest
 | 
	
		
			
				|  |  | -                    // possible size. The va_list::arg function will handle this
 | 
	
		
			
				|  |  | -                    // automagically.
 | 
	
		
			
				|  |  | -                    IntKind::Byte => ap.get(ArgKind::Byte, index).byte.to_string(),
 | 
	
		
			
				|  |  | -                    IntKind::Short => ap.get(ArgKind::Short, index).short.to_string(),
 | 
	
		
			
				|  |  | -                    // Types that will not be promoted
 | 
	
		
			
				|  |  | -                    IntKind::Int => ap.get(ArgKind::Int, index).int.to_string(),
 | 
	
		
			
				|  |  | -                    IntKind::Long => ap.get(ArgKind::Long, index).long.to_string(),
 | 
	
		
			
				|  |  | -                    IntKind::LongLong => ap.get(ArgKind::LongLong, index).longlong.to_string(),
 | 
	
		
			
				|  |  | -                    IntKind::PtrDiff => ap.get(ArgKind::PtrDiff, index).ptrdiff.to_string(),
 | 
	
		
			
				|  |  | -                    IntKind::Size => ap.get(ArgKind::Size, index).size.to_string(),
 | 
	
		
			
				|  |  | -                    IntKind::IntMax => ap.get(ArgKind::IntMax, index).intmax.to_string(),
 | 
	
		
			
				|  |  | +                let string = match varargs.get(index, &mut ap, Some(&arg)) {
 | 
	
		
			
				|  |  | +                    VaArg::c_char(i) => i.to_string(),
 | 
	
		
			
				|  |  | +                    VaArg::c_double(i) => panic!("this should not be possible"),
 | 
	
		
			
				|  |  | +                    VaArg::c_int(i) => i.to_string(),
 | 
	
		
			
				|  |  | +                    VaArg::c_long(i) => i.to_string(),
 | 
	
		
			
				|  |  | +                    VaArg::c_longlong(i) => i.to_string(),
 | 
	
		
			
				|  |  | +                    VaArg::c_short(i) => i.to_string(),
 | 
	
		
			
				|  |  | +                    VaArg::intmax_t(i) => i.to_string(),
 | 
	
		
			
				|  |  | +                    VaArg::pointer(i) => (i as usize).to_string(),
 | 
	
		
			
				|  |  | +                    VaArg::ptrdiff_t(i) => i.to_string(),
 | 
	
		
			
				|  |  | +                    VaArg::ssize_t(i) => i.to_string()
 | 
	
		
			
				|  |  |                  };
 | 
	
		
			
				|  |  |                  let positive = !string.starts_with('-');
 | 
	
		
			
				|  |  |                  let zero = precision == Some(0) && string == "0";
 | 
	
	
		
			
				|  | @@ -573,18 +582,17 @@ unsafe fn inner_printf<W: Write>(w: W, format: *const c_char, ap: va_list) -> io
 | 
	
		
			
				|  |  |                  pad(w, left, b' ', final_len..pad_space)?;
 | 
	
		
			
				|  |  |              },
 | 
	
		
			
				|  |  |              FmtKind::Unsigned => {
 | 
	
		
			
				|  |  | -                let string = match intkind {
 | 
	
		
			
				|  |  | -                    // va_list will promote the following two to a c_int
 | 
	
		
			
				|  |  | -                    IntKind::Byte => fmt_int(fmt, ap.get(ArgKind::Byte, index).byte),
 | 
	
		
			
				|  |  | -                    IntKind::Short => fmt_int(fmt, ap.get(ArgKind::Short, index).short),
 | 
	
		
			
				|  |  | -                    IntKind::Int => fmt_int(fmt, ap.get(ArgKind::Int, index).int),
 | 
	
		
			
				|  |  | -                    IntKind::Long => fmt_int(fmt, ap.get(ArgKind::Long, index).long),
 | 
	
		
			
				|  |  | -                    IntKind::LongLong => {
 | 
	
		
			
				|  |  | -                        fmt_int(fmt, ap.get(ArgKind::LongLong, index).longlong)
 | 
	
		
			
				|  |  | -                    }
 | 
	
		
			
				|  |  | -                    IntKind::PtrDiff => fmt_int(fmt, ap.get(ArgKind::PtrDiff, index).ptrdiff),
 | 
	
		
			
				|  |  | -                    IntKind::Size => fmt_int(fmt, ap.get(ArgKind::Size, index).size),
 | 
	
		
			
				|  |  | -                    IntKind::IntMax => fmt_int(fmt, ap.get(ArgKind::IntMax, index).intmax),
 | 
	
		
			
				|  |  | +                let string = match varargs.get(index, &mut ap, Some(&arg)) {
 | 
	
		
			
				|  |  | +                    VaArg::c_char(i) => fmt_int(fmt, i as c_uchar),
 | 
	
		
			
				|  |  | +                    VaArg::c_double(i) => panic!("this should not be possible"),
 | 
	
		
			
				|  |  | +                    VaArg::c_int(i) => fmt_int(fmt, i as c_uint),
 | 
	
		
			
				|  |  | +                    VaArg::c_long(i) => fmt_int(fmt, i as c_ulong),
 | 
	
		
			
				|  |  | +                    VaArg::c_longlong(i) => fmt_int(fmt, i as c_ulonglong),
 | 
	
		
			
				|  |  | +                    VaArg::c_short(i) => fmt_int(fmt, i as c_ushort),
 | 
	
		
			
				|  |  | +                    VaArg::intmax_t(i) => fmt_int(fmt, i as uintmax_t),
 | 
	
		
			
				|  |  | +                    VaArg::pointer(i) => fmt_int(fmt, i as usize),
 | 
	
		
			
				|  |  | +                    VaArg::ptrdiff_t(i) => fmt_int(fmt, i as size_t),
 | 
	
		
			
				|  |  | +                    VaArg::ssize_t(i) => fmt_int(fmt, i as size_t)
 | 
	
		
			
				|  |  |                  };
 | 
	
		
			
				|  |  |                  let zero = precision == Some(0) && string == "0";
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -628,20 +636,29 @@ unsafe fn inner_printf<W: Write>(w: W, format: *const c_char, ap: va_list) -> io
 | 
	
		
			
				|  |  |                  pad(w, left, b' ', final_len..pad_space)?;
 | 
	
		
			
				|  |  |              },
 | 
	
		
			
				|  |  |              FmtKind::Scientific => {
 | 
	
		
			
				|  |  | -                let mut float = ap.get(ArgKind::Double, index).double;
 | 
	
		
			
				|  |  | +                let mut float = match varargs.get(index, &mut ap, Some(&arg)) {
 | 
	
		
			
				|  |  | +                    VaArg::c_double(i) => i,
 | 
	
		
			
				|  |  | +                    _ => panic!("this should not be possible")
 | 
	
		
			
				|  |  | +                };
 | 
	
		
			
				|  |  |                  let precision = precision.unwrap_or(6);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |                  fmt_float_exp(w, fmt, None, false, precision, float, left, pad_space, pad_zero)?;
 | 
	
		
			
				|  |  |              },
 | 
	
		
			
				|  |  |              FmtKind::Decimal => {
 | 
	
		
			
				|  |  | -                let mut float = ap.get(ArgKind::Double, index).double;
 | 
	
		
			
				|  |  | +                let mut float = match varargs.get(index, &mut ap, Some(&arg)) {
 | 
	
		
			
				|  |  | +                    VaArg::c_double(i) => i,
 | 
	
		
			
				|  |  | +                    _ => panic!("this should not be possible")
 | 
	
		
			
				|  |  | +                };
 | 
	
		
			
				|  |  |                  let precision = precision.unwrap_or(6);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |                  fmt_float_normal(w, false, precision, float, left, pad_space, pad_zero)?;
 | 
	
		
			
				|  |  |              },
 | 
	
		
			
				|  |  |              FmtKind::AnyNotation => {
 | 
	
		
			
				|  |  | +                let mut float = match varargs.get(index, &mut ap, Some(&arg)) {
 | 
	
		
			
				|  |  | +                    VaArg::c_double(i) => i,
 | 
	
		
			
				|  |  | +                    _ => panic!("this should not be possible")
 | 
	
		
			
				|  |  | +                };
 | 
	
		
			
				|  |  |                  let exp_fmt = b'E' | (fmt & 32);
 | 
	
		
			
				|  |  | -                let mut float = ap.get(ArgKind::Double, index).double;
 | 
	
		
			
				|  |  |                  let precision = precision.unwrap_or(6);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |                  if !fmt_float_exp(
 | 
	
	
		
			
				|  | @@ -661,7 +678,10 @@ unsafe fn inner_printf<W: Write>(w: W, format: *const c_char, ap: va_list) -> io
 | 
	
		
			
				|  |  |              FmtKind::String => {
 | 
	
		
			
				|  |  |                  // if intkind == IntKind::Long || intkind == IntKind::LongLong, handle *const wchar_t
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -                let ptr = ap.get(ArgKind::CharPtr, index).char_ptr;
 | 
	
		
			
				|  |  | +                let mut ptr = match varargs.get(index, &mut ap, Some(&arg)) {
 | 
	
		
			
				|  |  | +                    VaArg::pointer(p) => p,
 | 
	
		
			
				|  |  | +                    _ => panic!("this should not be possible")
 | 
	
		
			
				|  |  | +                } as *const c_char;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |                  if ptr.is_null() {
 | 
	
		
			
				|  |  |                      w.write_all(b"(null)")?;
 | 
	
	
		
			
				|  | @@ -680,14 +700,20 @@ unsafe fn inner_printf<W: Write>(w: W, format: *const c_char, ap: va_list) -> io
 | 
	
		
			
				|  |  |              FmtKind::Char => {
 | 
	
		
			
				|  |  |                  // if intkind == IntKind::Long || intkind == IntKind::LongLong, handle wint_t
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -                let c = ap.get(ArgKind::Byte, index).byte;
 | 
	
		
			
				|  |  | +                let c = match varargs.get(index, &mut ap, Some(&arg)) {
 | 
	
		
			
				|  |  | +                    VaArg::c_char(c) => c,
 | 
	
		
			
				|  |  | +                    _ => panic!("this should not be possible")
 | 
	
		
			
				|  |  | +                };
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |                  pad(w, !left, b' ', 1..pad_space)?;
 | 
	
		
			
				|  |  |                  w.write_all(&[c as u8])?;
 | 
	
		
			
				|  |  |                  pad(w, left, b' ', 1..pad_space)?;
 | 
	
		
			
				|  |  |              },
 | 
	
		
			
				|  |  |              FmtKind::Pointer => {
 | 
	
		
			
				|  |  | -                let ptr = ap.get(ArgKind::VoidPtr, index).int_ptr;
 | 
	
		
			
				|  |  | +                let mut ptr = match varargs.get(index, &mut ap, Some(&arg)) {
 | 
	
		
			
				|  |  | +                    VaArg::pointer(p) => p,
 | 
	
		
			
				|  |  | +                    _ => panic!("this should not be possible")
 | 
	
		
			
				|  |  | +                };
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |                  let mut len = 1;
 | 
	
		
			
				|  |  |                  if ptr.is_null() {
 | 
	
	
		
			
				|  | @@ -709,14 +735,27 @@ unsafe fn inner_printf<W: Write>(w: W, format: *const c_char, ap: va_list) -> io
 | 
	
		
			
				|  |  |                  pad(w, left, b' ', len..pad_space)?;
 | 
	
		
			
				|  |  |              },
 | 
	
		
			
				|  |  |              FmtKind::GetWritten => {
 | 
	
		
			
				|  |  | -                let ptr = ap.get(ArgKind::IntPtr, index).int_ptr;
 | 
	
		
			
				|  |  | -                *ptr = w.written as c_int;
 | 
	
		
			
				|  |  | +                let mut ptr = match varargs.get(index, &mut ap, Some(&arg)) {
 | 
	
		
			
				|  |  | +                    VaArg::pointer(p) => p,
 | 
	
		
			
				|  |  | +                    _ => panic!("this should not be possible")
 | 
	
		
			
				|  |  | +                };
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +                match intkind {
 | 
	
		
			
				|  |  | +                    IntKind::Byte => *(ptr as *mut c_char) = w.written as c_char,
 | 
	
		
			
				|  |  | +                    IntKind::Short => *(ptr as *mut c_short) = w.written as c_short,
 | 
	
		
			
				|  |  | +                    IntKind::Int => *(ptr as *mut c_int) = w.written as c_int,
 | 
	
		
			
				|  |  | +                    IntKind::Long => *(ptr as *mut c_long) = w.written as c_long,
 | 
	
		
			
				|  |  | +                    IntKind::LongLong => *(ptr as *mut c_longlong) = w.written as c_longlong,
 | 
	
		
			
				|  |  | +                    IntKind::IntMax => *(ptr as *mut intmax_t) = w.written as intmax_t,
 | 
	
		
			
				|  |  | +                    IntKind::PtrDiff => *(ptr as *mut ptrdiff_t) = w.written as ptrdiff_t,
 | 
	
		
			
				|  |  | +                    IntKind::Size => *(ptr as *mut size_t) = w.written as size_t
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  |              }
 | 
	
		
			
				|  |  |          }
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  |      Ok(w.written as c_int)
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -pub unsafe fn printf<W: Write>(w: W, format: *const c_char, ap: va_list) -> c_int {
 | 
	
		
			
				|  |  | +pub unsafe fn printf<W: Write>(w: W, format: *const c_char, ap: VaList) -> c_int {
 | 
	
		
			
				|  |  |      inner_printf(w, format, ap).unwrap_or(-1)
 | 
	
		
			
				|  |  |  }
 |