#include <strformat.h>

#define HAVE_DOUBLE

#define HAVE_LONGLONG
#ifndef LARGEST_SIGNED
#ifdef HAVE_LONGLONG
#define LARGEST_SIGNED long long int
#else
#define LARGEST_UNSIGNED long int
#endif
#endif

#ifndef LARGEST_UNSIGNED
#ifdef HAVE_LONGLONG
#define LARGEST_UNSIGNED unsigned long long int
#else
#define LARGEST_UNSIGNED unsigned long int
#endif
#endif

#ifndef POINTER_INT
#define POINTER_INT unsigned long
#endif

typedef unsigned int FormatFlags;

#define MAKE_MASK(shift,size) (((1 << size) - 1) << (shift))

#define JUSTIFY_SHIFT	0
#define JUSTIFY_SIZE	1
#define JUSTIFY_RIGHT	0x0000
#define JUSTIFY_LEFT	0x0001
#define JUSTIFY_MASK	MAKE_MASK(JUSTIFY_SHIFT,JUSTIFY_SIZE)


/* How a positive number is prefixed */
#define POSITIVE_SHIFT	(JUSTIFY_SHIFT + JUSTIFY_SIZE)
#define POSITIVE_NONE	(0x0000 << POSITIVE_SHIFT)
#define POSITIVE_SPACE	(0x0001 << POSITIVE_SHIFT)
#define POSITIVE_PLUS	(0x0003 << POSITIVE_SHIFT)
#define POSITIVE_MASK	MAKE_MASK(POSITIVE_SHIFT, POSITIVE_SIZE)

#define POSITIVE_SIZE	2

#define ALTERNATE_FORM_SHIFT (POSITIVE_SHIFT + POSITIVE_SIZE)
#define ALTERNATE_FORM_SIZE 1
#define ALTERNATE_FORM  (0x0001 << ALTERNATE_FORM_SHIFT)


#define PAD_SHIFT	(ALTERNATE_FORM_SHIFT + ALTERNATE_FORM_SIZE)
#define PAD_SIZE	1
#define PAD_SPACE	(0x0000 << PAD_SHIFT)
#define PAD_ZERO	(0x0001 << PAD_SHIFT)

#define SIZE_SHIFT	(PAD_SHIFT + PAD_SIZE)
#define SIZE_SIZE	3
#define SIZE_CHAR	(0x0001 << SIZE_SHIFT)
#define SIZE_SHORT	(0x0002 << SIZE_SHIFT)
#define SIZE_INT	(0x0000 << SIZE_SHIFT)
#define SIZE_LONG	(0x0003 << SIZE_SHIFT)
#define SIZE_LONGLONG	(0x0004 << SIZE_SHIFT)
#define SIZE_MASK	MAKE_MASK(SIZE_SHIFT,SIZE_SIZE)

#define CONV_SHIFT	(SIZE_SHIFT + SIZE_SIZE)
#define CONV_SIZE	3
#define CONV_INTEGER	(0x0001 << CONV_SHIFT)
#define CONV_FLOAT	(0x0002 << CONV_SHIFT)
#define CONV_POINTER	(0x0003 << CONV_SHIFT)
#define CONV_STRING	(0x0004 << CONV_SHIFT)
#define CONV_CHAR	(0x0005 << CONV_SHIFT)
#define CONV_PERCENT	(0x0006 << CONV_SHIFT)
#define CONV_WRITTEN	(0x0007 << CONV_SHIFT)
#define CONV_MASK	MAKE_MASK(CONV_SHIFT, CONV_SIZE)

#define RADIX_SHIFT	(CONV_SHIFT + CONV_SIZE)
#define RADIX_SIZE	2
#define RADIX_DECIMAL	(0x0001 << RADIX_SHIFT)
#define RADIX_OCTAL	(0x0002 << RADIX_SHIFT)
#define RADIX_HEX	(0x0003 << RADIX_SHIFT)
#define RADIX_MASK	MAKE_MASK(RADIX_SHIFT,RADIX_SIZE)

#define SIGNED_SHIFT	(RADIX_SHIFT + RADIX_SIZE)
#define SIGNED_SIZE	1
#define SIGNED_NO	(0x0000 << SIGNED_SHIFT)
#define SIGNED_YES	(0x0001 << SIGNED_SHIFT)
#define SIGNED_MASK	MAKE_MASK(SIGNED_SHIFT,SIGNED_SIZE)

#define CAPS_SHIFT	(SIGNED_SHIFT + SIGNED_SIZE)
#define CAPS_SIZE	1
#define CAPS_NO		(0x0000 << CAPS_SHIFT)
#define CAPS_YES	(0x0001 << CAPS_SHIFT)
#define CAPS_MASK	MAKE_MASK(CAPS_SHIFT,CAPS_SIZE)

#define FLOAT_SHIFT	(CAPS_SHIFT + CAPS_SIZE)
#define FLOAT_SIZE	2
#define FLOAT_NORMAL	(0x0000 << FLOAT_SHIFT)
#define FLOAT_EXPONENT	(0x0001 << FLOAT_SHIFT)
#define FLOAT_DEPENDANT	(0x0002 << FLOAT_SHIFT)
#define FLOAT_HEX	(0x0003 << FLOAT_SHIFT)
#define FLOAT_MASK	MAKE_MASK(FLOAT_SHIFT, FLOAT_SIZE)

static FormatFlags
parse_flags(const char **posp)
{
  FormatFlags flags = 0;
  const char *pos = *posp;
  while (1) {
    switch(*pos) {
    case '-':
      flags |= JUSTIFY_LEFT;
      break;
    case '+':
      flags |= POSITIVE_PLUS;
      break;
    case ' ':
      flags |= POSITIVE_SPACE;
      break;
    case '#':
      flags |= ALTERNATE_FORM;
      break;
    case '0':
      flags |= PAD_ZERO;
      break;
    default:
      *posp = pos;
      return flags;
    }
    pos++;
  }
      
}

static unsigned int
parse_uint(const char **posp)
{
  unsigned v = 0;
  const char *pos = *posp;
  char ch;
  while((ch = *pos) >= '0' && ch <= '9') {
    v = v * 10 + (ch - '0');
    pos++;
  }
  *posp = pos;
  return v;
}

#define MAXCHARS_HEX ((sizeof(LARGEST_UNSIGNED) * 8) / 4 )

/* Largest number of characters needed for converting an unsigned integer.
 */
#define MAXCHARS ((sizeof(LARGEST_UNSIGNED) * 8  + 2) / 3 )

static unsigned int
output_uint_decimal(char **posp, LARGEST_UNSIGNED v)
{
  unsigned int len;
  char *pos = *posp;
  while (v > 0) {
    *--pos = (v % 10) + '0';
    v /= 10;
  }
  len = *posp - pos;
  *posp = pos;
  return len;
}

static unsigned int
output_uint_hex(char **posp, LARGEST_UNSIGNED v, unsigned int flags)
{
  unsigned int len;
  const char *hex = (flags & CAPS_YES) ?"0123456789ABCDEF":"0123456789abcdef";
  char *pos = *posp;
  while (v > 0) {
    *--pos = hex[(v % 16)];
    v /= 16;
  }
  len = *posp - pos;
  *posp = pos;
  return len;
}

static unsigned int
output_uint_octal(char **posp, LARGEST_UNSIGNED v)
{
  unsigned int len;
  char *pos = *posp;
  while (v > 0) {
    *--pos = (v % 8) + '0';
    v /= 8;
  }
  len = *posp - pos;
  *posp = pos;
  return len;
}

static StrFormatResult
fill_space(const StrFormatContext *ctxt, unsigned int len)
{
  StrFormatResult res;
  static const char buffer[16] = "                ";
  while(len > 16) {
    res = ctxt->write_str(ctxt->user_data, buffer, 16);
    if (res != STRFORMAT_OK) return res;
    len -= 16;
  }
  if (len == 0) return STRFORMAT_OK;
  return ctxt->write_str(ctxt->user_data, buffer, len);
}

static StrFormatResult
fill_zero(const StrFormatContext *ctxt, unsigned int len)
{
  StrFormatResult res;
  static const char buffer[16] = "0000000000000000";
  while(len > 16) {
    res = ctxt->write_str(ctxt->user_data, buffer, 16);
    if (res != STRFORMAT_OK) return res;
    len -= 16;
  }
  if (len == 0) return STRFORMAT_OK;
  return ctxt->write_str(ctxt->user_data, buffer, len);
}

#define CHECKCB(res) {if ((res) != STRFORMAT_OK) {va_end(ap); return -1;}}

int
format_str(const StrFormatContext *ctxt, const char *format, ...)
{
  int ret;
  va_list ap;
  va_start(ap, format);
  ret = format_str_v(ctxt, format, ap);
  va_end(ap);
  return ret;
}

int
format_str_v(const StrFormatContext *ctxt, const char *format, va_list ap)
{
  unsigned int written = 0;
  const char *pos = format;
  while(*pos != '\0') {
    FormatFlags flags;
    unsigned int minwidth = 0;
    int precision = -1; /* Negative means no precision */
    char ch;
    const char *start = pos;
    while( (ch = *pos) != '\0' && ch != '%') pos++;
    if (pos != start) {
      CHECKCB(ctxt->write_str(ctxt->user_data, start, pos - start));
      written += pos - start;
    }
    if (*pos == '\0') {
      va_end(ap);
      return written;
    }
    pos++;
    if (*pos == '\0') {
      va_end(ap);
      return written;
    }
    flags = parse_flags(&pos);

    /* parse width */
    if (*pos >= '1' && *pos <= '9') {
      minwidth = parse_uint(&pos);
    } else if (*pos == '*') {
      int w = va_arg(ap,int);
      if (w < 0) {
	flags |= JUSTIFY_LEFT;
	minwidth = w;
      } else {
	minwidth = w;
      }
      pos ++;
    }

    /* parse precision */
    if (*pos == '.') {
      pos++;
      if (*pos >= '0' && *pos <= '9') {
	precision = parse_uint(&pos);
      } else if (*pos == '*') {
	precision = va_arg(ap,int);
      }
    }
    if (*pos == 'l') {
      pos++;
      if (*pos == 'l') {
	flags |= SIZE_LONGLONG;
	pos++;
      } else {
	flags |= SIZE_LONG;
      }
    } else if (*pos == 'h') {
      pos++;
      if (*pos == 'h') {
	flags |= SIZE_CHAR;
	pos++;
      } else {
	flags |= SIZE_SHORT;
      }
    }

    /* parse conversion specifier */
    switch(*pos) {
    case 'd':
    case 'i':
      flags |= CONV_INTEGER | RADIX_DECIMAL | SIGNED_YES;
      break;
    case 'u':
      flags |= CONV_INTEGER | RADIX_DECIMAL | SIGNED_NO;
      break;
    case 'o':
      flags |= CONV_INTEGER | RADIX_OCTAL | SIGNED_NO;
      break;
    case 'x':
      flags |= CONV_INTEGER | RADIX_HEX | SIGNED_NO;
      break;
    case 'X':
      flags |= CONV_INTEGER | RADIX_HEX | SIGNED_NO | CAPS_YES;
      break;
#ifdef HAVE_DOUBLE
    case 'f':
      flags |= CONV_FLOAT | FLOAT_NORMAL;
      break;
    case 'F':
      flags |= CONV_FLOAT | FLOAT_NORMAL | CAPS_YES;
      break;
    case 'e':
      flags |= CONV_FLOAT | FLOAT_EXPONENT;
      break;
    case 'E':
      flags |= CONV_FLOAT | FLOAT_EXPONENT | CAPS_YES;
      break;
    case 'g':
      flags |= CONV_FLOAT | FLOAT_DEPENDANT;
      break;
    case 'G':
      flags |= CONV_FLOAT | FLOAT_DEPENDANT | CAPS_YES;
      break;
    case 'a':
      flags |= CONV_FLOAT | FLOAT_HEX;
      break;
    case 'A':
      flags |= CONV_FLOAT | FLOAT_HEX | CAPS_YES;
      break;
#endif
    case 'c':
      flags |= CONV_CHAR;
      break;
    case 's':
      flags |= CONV_STRING;
      break;
    case 'p':
      flags |= CONV_POINTER;
      break;
    case 'n':
      flags |= CONV_WRITTEN;
      break;
    case '%':
      flags |= CONV_PERCENT;
      break;
    case '\0':
      va_end(ap);
      return written;
    }
    pos++;
    switch(flags & CONV_MASK) {
    case CONV_PERCENT:
      CHECKCB(ctxt->write_str(ctxt->user_data, "%", 1));
      written++;
      break;
    case CONV_INTEGER:
      {
	/* unsigned integers */
	char *prefix = 0; /* sign, "0x" or "0X" */
	unsigned int prefix_len = 0;
	char buffer[MAXCHARS];
	char *conv_pos = buffer + MAXCHARS;
	unsigned int conv_len = 0;
	unsigned int width = 0;
	unsigned int precision_fill;
	unsigned int field_fill;
	LARGEST_UNSIGNED uvalue = 0;
	int negative = 0;
      
	if (precision < 0) precision = 1;
	else flags &= ~PAD_ZERO;
      
	if (flags & SIGNED_YES) {
	  /* signed integers */
	  LARGEST_SIGNED value = 0;
	  switch(flags & SIZE_MASK) {
	  case SIZE_CHAR:
	    value = (signed char)va_arg(ap, int);
	    break;
	  case SIZE_SHORT:
	    value = (short)va_arg(ap, int);
	    break;
	  case SIZE_INT:
	    value = va_arg(ap, int);
	    break;
#ifndef HAVE_LONGLONG
	  case SIZE_LONGLONG:	/* Treat long long the same as long */
#endif
	  case SIZE_LONG:
	    value = va_arg(ap, long);
	    break;
#ifdef HAVE_LONGLONG
	  case SIZE_LONGLONG:
	    value = va_arg(ap, long long);
	    break;
#endif
	  }
	  if (value < 0) {
	    uvalue = -value;
	    negative = 1;
	  } else {
	    uvalue = value;
	  }
	} else {
	
	  switch(flags & SIZE_MASK) {
	  case SIZE_CHAR:
	    uvalue = (unsigned char)va_arg(ap,unsigned int);
	    break;
	  case SIZE_SHORT:
	    uvalue = (unsigned short)va_arg(ap,unsigned int);
	    break;
	  case SIZE_INT:
	    uvalue = va_arg(ap,unsigned int);
	    break;
#ifndef HAVE_LONGLONG
	  case SIZE_LONGLONG:	/* Treat long long the same as long */
#endif
	  case SIZE_LONG:
	    uvalue = va_arg(ap,unsigned long);
	    break;
#ifdef HAVE_LONGLONG
	  case SIZE_LONGLONG:
	    uvalue = va_arg(ap,unsigned long long);
	    break;
#endif
	  }
	}
	
	switch(flags & (RADIX_MASK)) {
	case RADIX_DECIMAL:
	  conv_len = output_uint_decimal(&conv_pos,uvalue);
	  break;
	case RADIX_OCTAL:
	  conv_len = output_uint_octal(&conv_pos,uvalue);
	  break;
	case RADIX_HEX:
	  conv_len = output_uint_hex(&conv_pos,uvalue, flags);
	  break;
	}

	width += conv_len;
	precision_fill = (precision > conv_len) ? precision - conv_len : 0;
	if ((flags & (RADIX_MASK | ALTERNATE_FORM))
	    == (RADIX_OCTAL | ALTERNATE_FORM)) {
	  if (precision_fill < 1) precision_fill = 1;
	}

	width += precision_fill;
	
	if ((flags & (RADIX_MASK | ALTERNATE_FORM))
	    == (RADIX_HEX | ALTERNATE_FORM) && uvalue != 0) {
	  prefix_len = 2;
	  if (flags & CAPS_YES) {
	    prefix = "0X";
	  } else {
	    prefix = "0x";
	  }
	}

	if (flags & SIGNED_YES) {
	  if (negative) {
	    prefix = "-";
	    prefix_len = 1;
	  } else {
	    switch(flags & POSITIVE_MASK) {
	    case POSITIVE_SPACE:
	      prefix = " ";
	      prefix_len = 1;
	      break;
	    case POSITIVE_PLUS:
	      prefix = "+";
	      prefix_len = 1;
	      break;
	    }
	  }
	}

	width += prefix_len;

	field_fill = (minwidth > width) ? minwidth - width : 0;

	if ((flags & JUSTIFY_MASK) == JUSTIFY_RIGHT) {
	  if (flags & PAD_ZERO) {
	    precision_fill += field_fill;
	  } else {
	    CHECKCB(fill_space(ctxt,field_fill));
	  }
	}

	if (prefix_len > 0)
	  CHECKCB(ctxt->write_str(ctxt->user_data, prefix, prefix_len));
	written += prefix_len;
	
	CHECKCB(fill_zero(ctxt,precision_fill));
	written += prefix_len;
	
	CHECKCB(ctxt->write_str(ctxt->user_data, conv_pos,conv_len));
	written += conv_len;
	
	if ((flags & JUSTIFY_MASK) == JUSTIFY_LEFT) {
	  CHECKCB(fill_space(ctxt,field_fill));
	}
	written += field_fill;
      }
      break;
    case CONV_STRING:
      {
	unsigned int field_fill;
	unsigned int len;
	char *str = va_arg(ap,char *);
	if (str) {
	  char *pos = str;
	  while(*pos != '\0') pos++;
	  len = pos - str;
	} else {
	  str = "(null)";
	  len = 6;
	}
	if (precision >= 0 && precision < len) len = precision;
	field_fill = (minwidth > len) ? minwidth - len : 0;
	if ((flags & JUSTIFY_MASK) == JUSTIFY_RIGHT) {
	  CHECKCB(fill_space(ctxt,field_fill));
	}
	CHECKCB(ctxt->write_str(ctxt->user_data, str,len));
	written += len;
	if ((flags & JUSTIFY_MASK) == JUSTIFY_LEFT) {
	  CHECKCB(fill_space(ctxt,field_fill));
	}
	written += field_fill;
      }
      break;
    case CONV_POINTER:
      {
	LARGEST_UNSIGNED uvalue =
	  (LARGEST_UNSIGNED)(POINTER_INT)va_arg(ap,void *);
	char buffer[MAXCHARS_HEX + 3];
	char *conv_pos = buffer + MAXCHARS_HEX+3;
	unsigned int conv_len;
	unsigned int field_fill;
	
	conv_len = output_uint_hex(&conv_pos,uvalue,flags);
	if (conv_len == 0) {
	  *--conv_pos = '0';
	  conv_len++;
	}
	*--conv_pos = 'x';
	*--conv_pos = '0';
	*--conv_pos = '#';
	conv_len += 3;

	field_fill = (minwidth > conv_len) ? minwidth - conv_len : 0;
	
	if ((flags & JUSTIFY_MASK) == JUSTIFY_RIGHT) {
	  CHECKCB(fill_space(ctxt,field_fill));
	}
	
	CHECKCB(ctxt->write_str(ctxt->user_data, conv_pos,conv_len));
	written += conv_len;
	
	if ((flags & JUSTIFY_MASK) == JUSTIFY_LEFT) {
	  CHECKCB(fill_space(ctxt,field_fill));
	}
	written += field_fill;
      }
      break;
    case CONV_CHAR:
      {
	char ch = va_arg(ap,int);
	unsigned int field_fill = (minwidth > 1) ? minwidth - 1 : 0;
	if ((flags & JUSTIFY_MASK) == JUSTIFY_RIGHT) {
	  CHECKCB(fill_space(ctxt,field_fill));
	  written += field_fill;
	}
	
	CHECKCB(ctxt->write_str(ctxt->user_data, &ch, 1));
	written++;
	
	if ((flags & JUSTIFY_MASK) == JUSTIFY_LEFT) {
	  CHECKCB(fill_space(ctxt,field_fill));
	}
	written+= field_fill;
      }
      break;
    case CONV_WRITTEN:
      {
	int *p = va_arg(ap,int*);
	*p = written;
      }
      break;

    }
  }
  
  return written;
}