HP85 GPIB Disk Emulator  1.0
HP85GPIBDiskEmulator
All Data Structures Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
printf.c
Go to the documentation of this file.
1 
28 #ifdef USER_CONFIG
29 #include "user_config.h"
30 #else
31 //#include <stdio.h>
32 #endif
33 
34 #include <stdint.h>
35 #include <stdarg.h>
36 #include <string.h>
37 #include <math.h>
38 
39 #include "mathio.h"
40 
41 // =============================================
42 
43 // Below we included functions that are defined elsewhere
44 // They are included within printf to allow this to be standalone
45 // This is done by looking at header defines for the time being
46 
51 size_t
53 strlen(const char *str)
54 {
55  int len=0;
56 // String length
57  while(*str++)
58  ++len;
59  return(len);
60 }
61 
62 
63 // Skip if we have linux ctype.h
67 #undef isdigit
69 int
71 isdigit(int c)
72 {
73  if(c >= '0' && c <= '9')
74  return(1);
75  return(0);
76 }
77 
78 
79 // =============================================
80 // start of support functions
81 // =============================================
82 
88 void
90 reverse(char *str)
91 {
92  char temp;
93  int i;
94  int len = strlen(str);
95 // Reverse
96 // We only exchange up to half way
97  for (i = 0; i < (len >> 1); i++)
98  {
99  temp = str[len - i - 1];
100  str[len - i - 1] = str[i];
101  str[i] = temp;
102  }
103 }
104 
105 
109 MEMSPACE
110 void
111 WEAK_ATR
112 strupper(char *str)
113 {
114  while(*str)
115  {
116  if(*str >= 'a' && *str <= 'z')
117  *str -= ( 'a' - 'A');
118  ++str;
119  }
120 }
121 
122 
123 #if __BYTE_ORDER__ != __ORDER_LITTLE_ENDIAN__
124 #error bin2num ASSUME little endian
125 #endif
126 
142 MEMSPACE
143 int bin2num(uint8_t *str, int strmax, int nummin, int base, uint8_t *nump, int numsize, int sign_ch)
144 {
145  int i,j,carry;
146  uint8_t data;
147 
148  int numbits = numsize * 8;
149 
150  for(i=0;i<=nummin;++i)
151  str[i] = 0; // initial string starts out empty
152 
153 // FIXME Little/Big-endian
154 // Loop for all bits (numsize in total) in the binary number (bin)
155 // Examine each bit MSB to LSB order
156  for(i = numbits - 1; i>= 0; --i)
157  {
158 // We extract 1 bit at a time from the binary number (MSB to LSB order)
159 // FIXME Little/Big-endian
160  data = nump[i>>3];
161 // If extracted bit was set carry = 1, else 0
162 // FIXME Little/Big-endian
163  carry = ( data & (1 << (i & 7)) ) ? 1 : 0;
164 
165 // Multiply base number by two and add the previously extracted bit
166 // Next do base digit to digit carries as needed
167 // Note: nummin is the current string size that can grow as needed
168 // Carry test in the loop can optionally extend base strings size
169  for(j=0;(j<nummin || carry) ;++j)
170  {
171  if(j >= (strmax - 2))
172  break;
173 
174  data = str[j];
175  data = (data<<1) | carry;
176 // process base carry
177  carry = 0;
178  if(data >= base)
179  {
180  data -= base;
181  carry = 1;
182  }
183  str[j] = data;
184  }
185  str[j] = 0; // zero next digit if carry extends size
186  nummin = j; // update nummin if carry extends size
187  }
188 
189 // Add ASCII '0' or 'a' offsets to base string
190  for(i=0;i<nummin;++i)
191  {
192  if(str[i] < 10)
193  str[i] += '0';
194  else str[i] += 'a'-10;
195  }
196 
197 // Add optional sign character
198  if(sign_ch && i <= (strmax - 2))
199  {
200  str[i++] = sign_ch;
201  ++nummin;
202  }
203  str[i] = 0; // Terminate string eith EOS
204 
205  reverse((char *)str); // Reverse in place to correct order
206 
207  return(nummin); // Return string size
208 }
209 
210 
211 // =============================================
212 
217 MEMSPACE
218 void pch_init(p_ch_t *p, char *str, int max)
219 {
220  p->str = str;
221  p->ind = 0;
222  p->max = max-1;
223  p->str[0] = 0;
224 }
225 
226 
232 MEMSPACE
233 int pch(p_ch_t *p, char ch)
234 {
235 // Add the character while ther is room
236  if(p->ind < p->max)
237  p->str[p->ind++] = ch;
238  else
239  p->str[p->ind] = 0; // Add EOS when limit exceeded
240  return(p->ind);
241 }
242 
243 
246 MEMSPACE
248 {
249  return(p->ind);
250 }
251 
252 
255 MEMSPACE
257 {
258  return(p->max);
259 }
260 
261 
262 // =============================================
263 // end of support functions
264 // =============================================
268 MEMSPACE
270 {
271  if(f.b.left)
272  printf("left flag\n");
273  if(f.b.plus)
274  printf("plus flag\n");
275  if(f.b.space)
276  printf("space flag\n");
277  if(f.b.zero)
278  printf("zero flag\n");
279  if(f.b.alt)
280  printf("alt flag\n");
281  if(f.b.width)
282  printf("width flag\n");
283  if(f.b.prec)
284  printf("prec flag\n");
285  if(f.b.neg)
286  printf("< 0 flag\n");
287 }
288 
289 
312 
313 MEMSPACE
314 int p_ntoa(uint8_t *nump, int numsize, char *str, int strmax, int radix, int width, int prec, f_t f)
315 {
316  unsigned int sign_ch;
317  int ind;
318  int digits;
319 
320  digits = 0;
321 
322 // Unsigned, hex,octal,binary should leave these flasg zero
323 //f.b.space = 0;
324 //f.b.plus = 0;
325 //f.b.neg = 0;
326 
327  sign_ch = 0;
328  if(f.b.neg)
329  sign_ch = '-';
330  else if(f.b.plus)
331  sign_ch = '+';
332  else if(f.b.space)
333  sign_ch = ' ';
334 
335 //print_flags(f);
336 
337 /* Some Combinations of flags are not permitted
338  * - or impact the interpretation of others */
339  if(f.b.zero)
340  {
341 // 0 disabled if precision or left align
342  if(f.b.left || f.b.prec)
343  f.b.zero = 0;
344  }
345 
346  if(f.b.prec)
347  digits = prec;
348 
349  if(!f.b.width && !f.b.prec)
350  digits = 1; // at least one digit
351 
352  if(f.b.width)
353  {
354  if(!f.b.zero)
355  {
356 // Width and no leading zeros require at least one digit
357  if(!f.b.prec)
358  digits = 1; // at least one digit
359  }
360  else /* precision and 0 can not occur together - previously tested */
361  {
362  digits = width;
363 /* make room for a sign ? */
364  if(f.b.plus || f.b.neg || f.b.space)
365  --digits;
366  }
367  }
368  ind = bin2num((uint8_t *)str, strmax, digits, radix, nump, numsize, sign_ch);
369  return(ind);
370 }
371 
372 
373 #ifdef FLOATIO
374 MEMSPACE
382 int p_ftoa(double val, char *str, int max, int width, int prec, f_t f)
383 {
384 
385  char *save = str;
386  double fscale;
387  int idigits, digits;
388  int digit;
389  p_ch_t _pch;
390  p_ch_t *p = (p_ch_t *) &_pch;
391  pch_init(p, str,max);
392 
393 /*
394 FIXME
395 Notice rounding - testing will be tricky
396 ERROR: [% 15.1f], [-10252956608208.250000]
397  G[-10252956608208.2]
398  B[-10252956608208.3]
399  error:9.90567929632045986708e-15
400 
401 */
402 //printf("val:%.20e\n", val);
403 //print_flags(f);
404  if(val < 0.0)
405  {
406  val = -val;
407  f.b.neg = 1;
408  }
409  if(f.b.neg)
410  pch(p,'-');
411  else if(f.b.plus)
412  pch(p,'+');
413  else if(f.b.space)
414  pch(p,' ');
415 
416 // prec only applies to fractional digits
417  if(prec < 0)
418  prec = 0;
419 
420 // NOTE: prec is anchored at the decimal point
421 
422  fscale = 0.0;
423  idigits = 1; // number of integer digits
424  if(val) // if zero no rounding needed
425  {
426 // do rounding - IF precision specified
427 // number of fractional digits to display
428  fscale = 0.5; // rounding value if prec = 0
429  if(f.b.prec)
430  {
431  digits = prec;
432  while(digits > 0)
433  {
434  fscale /= 10.0; // adjust rounding value
435  --digits;
436  }
437  }
438 //printf("val;:%.20e, fscale:%.20e\n", val, fscale);
439  val += fscale; // round to prec
440 
441  while(val >= 10.0)
442  {
443  ++idigits;
444  val /= 10.0;
445  }
446 // So we know that fval < 1.0;
447  }
448 
449 //printf("idigits:%d\n",idigits);
450 
451  if(f.b.zero && !f.b.left)
452  {
453  if(f.b.prec && prec)
454  digits = width - idigits - pch_ind(p) - prec -1;
455  else
456  digits = width - idigits - pch_ind(p);
457 
458  while(digits > 0)
459  {
460  pch(p,'0');
461  --digits;
462  }
463  }
464 
465 // Display integer part of number
466  while(idigits > 0)
467  {
468  digit = val;
469 //printf("ival:%.16e, int:%d\n", ival, digit);
470  pch(p,digit + '0');
471  val -= (double) digit;
472  --idigits;
473  val *= 10.0;
474  }
475 // display fractional part
476  if(f.b.prec && prec > 0 )
477  {
478  pch(p,'.');
479  while(prec > 0 )
480  {
481  digit = val;
482  val -= (double) digit;
483  digit += '0';
484  pch(p,digit);
485  --prec;
486  val *= 10.0;
487  }
488  }
489  pch(p,0);
490  return(strlen(save));
491 }
492 
493 
500 MEMSPACE
501 int p_etoa(double val,char *str, int max, int width, int prec, f_t f)
502 {
503  double fscale;
504  int digits;
505  int digit;
506  int exp10;
507  uint8_t exp10_str[7]; // +E123 and EOS
508  int expsize;
509  int i;
510  int sign_ch;
511  p_ch_t _pch;
512  p_ch_t *p = (p_ch_t *) &_pch;
513  pch_init(p, str,max);
514 
515 /* FIXME
516 Notice rounding - testing will be tricky
517 ERROR: [% 15.1f], [-10252956608208.250000]
518  G[-10252956608208.2]
519  B[-10252956608208.3]
520  error:9.90567929632045986708e-15
521 
522 */
523 //printf("val:%.20e\n", val);
524 //print_flags(f);
525  if(val < 0.0)
526  {
527  val = -val;
528  f.b.neg = 1;
529  }
530  if(f.b.neg)
531  pch(p,'-');
532  else if(f.b.plus)
533  pch(p,'+');
534  else if(f.b.space)
535  pch(p,' ');
536 
537 // prec only applies to fractional digits
538  if(prec < 0)
539  prec = 0;
540 
541 // NOTE: prec is anchored at the decimal point
542 
543  fscale = 1.0; // rounding value if prec = 0
544  exp10 = 0;
545 
546  if(val) // if zero no rounding needed
547  {
548  fscale = 0.5; // rounding value if prec = 0
549 // do rounding - IF precision specified
550 // number of fractional digits to display
551  if(f.b.prec)
552  {
553  digits = prec;
554  while(digits > 0)
555  {
556  fscale /= 10.0; // adjust rounding value
557  --digits;
558  }
559  }
560 
561 // scale number 1.0 <= x < 10.0
562  while(val < 1.0 )
563  {
564  val *= 10.0;
565  exp10--;
566 
567  }
568  while(val >= 10.0)
569  {
570  ++exp10;
571  val /= 10.0;
572  }
573  val += fscale;
574  while(val >= 10.0)
575  {
576  ++exp10;
577  val /= 10.0;
578  }
579 // round
580 //printf("val;:%.20e, fscale:%.20e\n", val, fscale);
581 //val += fscale; // round to prec
582  }
583 
584 // ====================
585 // Exponent
586  exp10_str[0] = 'e';
587  if ( exp10 < 0 )
588  {
589  sign_ch = '-';
590  exp10 = -exp10;
591  }
592  else
593  {
594  sign_ch = '+';
595  }
596 // result is +NN[N] if we have three digits shorten digits by one
597  expsize = bin2num(exp10_str+1, sizeof(exp10_str)-1-1, 2, 10, (uint8_t *) &exp10, sizeof(int), sign_ch);
598 
599 #if 0
600  if ( exp10 < 0 )
601  {
602  exp10_str[1] = '-';
603  exp10 = -exp10;
604  }
605  else
606  {
607  exp10_str[1] = '+';
608  }
609  if(exp >= 100)
610  ind = 4;
611  else
612  ind = 3;
613 
614  exp10_str[ind] = '0' + (exp % 10);
615  exp10 /= 10;
616  exp10_str[ind] = '0' + (exp % 10);
617  exp10 /= 10;
618  exp10_str[ind] = '0' + (exp % 10);
619 #endif
620 
621 // ====================
622 
623 // [+]N.FFe+00, where ".e+00" = 5 digits, pch_ind holds optional sign offset
624  if(f.b.zero && !f.b.left)
625  {
626  if(f.b.prec && prec)
627  digits = width - pch_ind(p) - prec - 6;
628  else
629  digits = width - pch_ind(p) - 5;
630  if(expsize > 3)
631  --digits;
632  while(digits > 0)
633  {
634  pch(p,'0');
635  --digits;
636  }
637  }
638 
639 // Number
640  digit = val;
641 //printf("ival:%.16e, int:%d\n", ival, digit);
642  pch(p,digit + '0');
643  val -= (double) digit;
644  val *= 10.0;
645 
646 // display fractional part
647  if(f.b.prec && prec > 0 )
648  {
649  pch(p,'.');
650  while(prec > 0 )
651  {
652  digit = val;
653  val -= (double) digit;
654  digit += '0';
655  pch(p,digit);
656  val *= 10.0;
657  --prec;
658  }
659  }
660 
661  for(i=0;exp10_str[i];++i)
662  pch(p,exp10_str[i]);
663 
664  pch(p,0);
665  return(strlen(str));
666 }
667 #endif // FLOATIO
668 
669 
670 // =============================================
677 MEMSPACE
678 void _puts_pad(printf_t *fn, char *s, int width, int count, int left)
679 {
680  int size = 0;
681  int pad = 0;
682 
683 // note - if width > count we pad
684 // if width <= count we do not pad
685  if(width > count)
686  {
687  pad = width - count;
688  }
689 
690 //printf("_puts_pad:(%s) width:%d, count:%d, left:%d, pad:%d, len:%d\n", s, width, count, left, pad, len);
691 
692 // left padding ?
693  if(!left)
694  {
695 //printf("_puts_pad:pad:%d\n", pad);
696  while(pad--)
697  {
698  fn->put(fn,' ');
699  ++size;
700  }
701  }
702 //printf("_puts_pad:count:%d\n", count);
703 
704 // string
705  while(*s && count--)
706  {
707  fn->put(fn,*s);
708  ++s;
709  ++size;
710  }
711 // right padding
712  if(left)
713  {
714 //printf("_puts_pad:pad:%d\n", pad);
715 
716  while(pad--)
717  {
718  fn->put(fn,' ');
719  ++size;
720  }
721  }
722 //printf("_puts_pad:size:%d\n", size);
723 } // _puts_pad()
724 
725 #ifdef AVR
726 // =============================================
734 MEMSPACE
735 void _puts_pad_X(printf_t *fn, __memx const char *s, int width, int count, int left)
736 {
737  int size = 0;
738  int pad = 0;
739 
740 // note - if width > count we pad
741 // if width <= count we do not pad
742  if(width > count)
743  {
744  pad = width - count;
745  }
746 
747 //printf("_puts_pad:(%s) width:%d, count:%d, left:%d, pad:%d, len:%d\n", s, width, count, left, pad, len);
748 
749 // left padding ?
750  if(!left)
751  {
752 //printf("_puts_pad:pad:%d\n", pad);
753  while(pad--)
754  {
755  fn->put(fn,' ');
756  ++size;
757  }
758  }
759 //printf("_puts_pad:count:%d\n", count);
760 
761 // string
762  while(*s && count--)
763  {
764  fn->put(fn,*s);
765  ++s;
766  ++size;
767  }
768 // right padding
769  if(left)
770  {
771 //printf("_puts_pad:pad:%d\n", pad);
772 
773  while(pad--)
774  {
775  fn->put(fn,' ');
776  ++size;
777  }
778  }
779 //printf("_puts_pad:size:%d\n", size);
780 } // _puts_pad()
781 
785 MEMSPACE
786 size_t
787 WEAK_ATR
788 strlen_X(__memx const char *str)
789 {
790  int len=0;
791 // String length
792  while(*str++)
793  ++len;
794  return(len);
795 }
796 
797 #endif
798 
799 
805 MEMSPACE
806 void _printf_fn(printf_t *fn, __memx const char *fmt, va_list va)
807 {
808  int prec, width;
809  int count;
810  int spec;
811  int size;
812  int sign;
813  short nums;
814  int numi;
815  long numl;
816  long long numll;
817  void * numv;
818 #ifdef __SIZEOF_INT128__
819  __uint128_t num128;
820 #endif
821  uint8_t *nump;
822 
823  f_t f;
824 #ifdef FLOATIO
825  double dnum = 0;
826 #endif
827  char chartmp[2];
828  char *ptr;
829  __memx const char *fmtptr;
830  __memx const char *Xptr;
831 
832 
833 // buff has to be at least as big at the largest converted number
834 // in this case base 2 long long with sign and end of string
835 #ifdef PRINTF_TEST
836  char buff[307 * 2 + 5 + 1]; // double to full size
837 #else
838  char buff[sizeof( long long ) * 8 + 2];
839 #endif
840 
841  while(*fmt)
842  {
843 // emit up to %
844  if(*fmt != '%')
845  {
846  fn->put(fn,*fmt++);
847  continue;
848  }
849 
850  fmtptr = fmt;
851 // process % specifier
852  fmt++;
853 
854  prec = 0; // minimum number of digits displayed
855  width = 0; // padded width
856 
857 // we accept multiple flag combinations and duplicates as does GLIBC printf
858 // ['#']['-'][' '|'+']
859 // [' '|'+']['-']['#']
860 // ...
861 
862 // reset flags
863  f.all = 0;
864  while(*fmt == '#' || *fmt == '+' || *fmt == '-' || *fmt == ' ' || *fmt == '0')
865  {
866  if(*fmt == '#')
867  f.b.alt = 1;
868  else if(*fmt == '+')
869  f.b.plus = 1;
870  else if(!f.b.left && *fmt == '-')
871  f.b.left = 1;
872  else if(!f.b.space && *fmt == ' ')
873  f.b.space = 1;
874  else if(!f.b.zero && *fmt == '0')
875  f.b.zero = 1;
876 // format error
877  ++fmt;
878  }
879 
880 // width specifier
881 // Note: we permit zero as the first digit
882  if(isdigit(*fmt))
883  {
884 // optional width
885  width = 0;
886  while(isdigit(*fmt))
887  width = width*10 + *fmt++ - '0';
888  f.b.width = 1;
889  }
890 
891 // prec always impiles zero fill to prec digigits for ints and longs
892 // is the number of digits after the . for float and double
893 // regardlles of sign
894  if( *fmt == '.' )
895  {
896  fmt++;
897  prec = 0;
898  while(isdigit(*fmt))
899  prec = prec*10 + *fmt++ - '0';
900  f.b.prec = 1;
901  }
902 
908  size = sizeof(int); // int is default
909 
910  if( *fmt == 'I' )
911  {
912  fmt++;
913  size = 0;
914  while(isdigit(*fmt))
915  size = size*10 + *fmt++ - '0';
916  if(size == 0 || size & 7)
917  size = 0;
918  else
919  size >>= 3;
920  }
921  else if(*fmt == 'h')
922  {
923  fmt++;
924  size = sizeof(short);
925  }
926  else if(*fmt == 'l')
927  {
928  fmt++;
929  size = sizeof(long);
930  if(*fmt == 'l')
931  {
932  fmt++;
933  size = sizeof(long long);
934  }
935  }
936 
937  if(size)
938  spec = *fmt;
939  else
940  spec = 0;
941 
942  sign = 0;
943  if(spec == 'd' || spec == 'D')
944  sign = 1;
945 
946  nump = (uint8_t *) &numi;
947 
948 // process integer arguments
949  switch(spec)
950  {
951  case 'p':
952  case 'P':
953  size = sizeof(void *);
954 // Unsigned numbers
955  case 'b':
956  case 'B':
957  case 'o':
958  case 'O':
959  case 'x':
960  case 'X':
961  if(f.b.zero && f.b.left)
962  f.b.zero = 0;
963  if(f.b.zero && f.b.prec)
964  f.b.zero = 0;
965  if(f.b.zero && f.b.width)
966  {
967  if(width > prec)
968  prec = width;
969  }
970  if(f.b.zero && f.b.width && f.b.prec)
971  {
972  if(width > prec)
973  prec = width;
974  }
975  case 'u':
976  case 'U':
977  f.b.space = 0;
978  f.b.plus = 0;
979  f.b.neg = 0;
980  case 'D':
981  case 'd':
982 // make lint shut up
983 //FIXME vararg functions promote short - make this a conditional
984  if(size == sizeof(short))
985  {
986  nums = (short) va_arg(va, int);
987  if(sign && nums < 0)
988  {
989  f.b.neg = 1;
990  nums = -nums;
991  }
992  nump = (uint8_t *) &nums;
993  }
994  else if(size == sizeof(int))
995  {
996  numi = (int) va_arg(va, int);
997  if(sign && numi < 0)
998  {
999  f.b.neg = 1;
1000  numi = -numi;
1001  }
1002  nump = (uint8_t *) &numi;
1003  }
1004  else if(size == sizeof(long))
1005  {
1006  numl = (long) va_arg(va, long);
1007  if(sign && numl < 0)
1008  {
1009  f.b.neg = 1;
1010  numl = -numl;
1011  }
1012  nump = (uint8_t *) &numl;
1013  }
1014  else if(size == sizeof(long long))
1015  {
1016  numll = (long long) va_arg(va, long long);
1017  if(sign && numll < 0)
1018  {
1019  f.b.neg = 1;
1020  numll = -numll;
1021  }
1022  nump = (uint8_t *) &numll;
1023  }
1024 #ifdef __SIZEOF_INT128__
1025  else if(size == sizeof(__uint128_t))
1026  {
1027  num128 = (__uint128_t) va_arg(va, __uint128_t);
1028  if(sign && numll < 0)
1029  {
1030  f.b.neg = 1;
1031  num128 = -128;
1032  }
1033  nump = (uint8_t *) &num128;
1034  }
1035 #endif
1036  else if(size == sizeof(void *))
1037  {
1038  numv = (void *) va_arg(va, void *);
1039  nump = (uint8_t *) &numv;
1040  }
1041  else
1042  {
1043  spec = 0;
1044  }
1045 // FIXME default;
1046  ++fmt;
1047  break;
1048 #ifdef FLOATIO
1049  case 'f':
1050  case 'F':
1051  case 'e':
1052  case 'E':
1053 // K&R defines 'f' type as 6 - and matches GNU printf
1054  if(!f.b.prec)
1055  {
1056  prec = 6;
1057  f.b.prec = 1;
1058  }
1059  dnum = va_arg(va, double);
1060  ++fmt;
1061  break;
1062 #endif
1063  case 's':
1064  ++fmt;
1065  break;
1066  case 'S':
1067  ++fmt;
1068  break;
1069  case 'c':
1070  ++fmt;
1071  break;
1072  default:
1073  break;
1074  }
1075 
1076  switch(spec)
1077  {
1078  case 'u':
1079  case 'U':
1080  f.b.space = 0;
1081  f.b.plus = 0;
1082 // FIXME sign vs FILL
1083 //count = p_itoa(nump, size, buff, sizeof(buff), width, prec, f);
1084  count = p_ntoa(nump, size, buff, sizeof(buff), 10, width, prec, f);
1085  _puts_pad(fn,buff, width, count, f.b.left);
1086  break;
1087 // FIXME sign vs FILL
1088  case 'd':
1089  case 'D':
1090  count = p_ntoa(nump, size, buff, sizeof(buff), 10, width, prec, f);
1091  _puts_pad(fn,buff, width, count, f.b.left);
1092  break;
1093  case 'b':
1094  case 'B':
1095  count = p_ntoa(nump, size, buff, sizeof(buff), 2, width, prec,f);
1096  _puts_pad(fn,buff, width, count, f.b.left);
1097  break;
1098  case 'o':
1099  case 'O':
1100  count = p_ntoa(nump, size, buff, sizeof(buff), 8, width, prec,f);
1101  _puts_pad(fn,buff, width, count, f.b.left);
1102  break;
1103  case 'p':
1104  case 'P':
1105 // size = sizeof(void *);
1106  case 'x':
1107  case 'X':
1108  count = p_ntoa(nump, size, buff, sizeof(buff), 16, width, prec,f);
1109  if(spec == 'X' || spec == 'P')
1110  strupper(buff);
1111  _puts_pad(fn,buff, width, count, f.b.left);
1112  break;
1113 #ifdef FLOATIO
1114  case 'f':
1115  case 'F':
1116  count = p_ftoa(dnum, buff, sizeof(buff), width, prec, f);
1117  _puts_pad(fn,buff, width, count, f.b.left);
1118  break;
1119 
1120  case 'e':
1121  case 'E':
1122  count = p_etoa(dnum, buff, sizeof(buff), width, prec, f);
1123  if(spec == 'E')
1124  strupper(buff);
1125  _puts_pad(fn,buff, width, count, f.b.left);
1126  break;
1127 #endif
1128  case 's':
1129  case 'c':
1130  ptr = NULL; // stops bogus error that ptr may be uninitalized
1131 
1132  if(spec == 's')
1133  {
1134  ptr = va_arg(va, char *);
1135  if(!ptr)
1136  ptr = "(null)";
1137  }
1138  else // 'c'
1139  {
1140  chartmp[0] = (char) va_arg(va, int);
1141  chartmp[1] = 0;
1142  ptr = chartmp;
1143  }
1144  count = strlen(ptr);
1145  if(prec)
1146  count = prec;
1147  if(count > width && width != 0)
1148  count = width;
1149 //printf("width:%d,count:%d,left:%d\n", width, count, f.b.left);
1150  _puts_pad(fn, ptr, width, count, f.b.left);
1151  break;
1152 // Added string in program memory - testing
1153 #ifdef AVR
1154  case 'S':
1155  // Xptr = va_arg(va, __memx const char *);
1156  Xptr = va_arg(va, __flash const char *);
1157  if(!Xptr)
1158  Xptr = (__memx const char *) "(null)";
1159  count = strlen_X(Xptr);
1160  if(prec)
1161  count = prec;
1162  if(count > width && width != 0)
1163  count = width;
1164  _puts_pad_X(fn, Xptr, width, count, f.b.left);
1165  break;
1166 #endif
1167  default:
1168  while(fmtptr <= fmt && *fmtptr)
1169  fn->put(fn, *fmtptr++);
1170  break;
1171  }
1172 //printf("fmt:(%s)\n", fmt);
1173  }
1174 //printf("fmt exit:(%s)\n", fmt);
1175 
1176 }
1177 
1178 
1179 // =============================================
1186 MEMSPACE
1187 void _putc_buffer_fn(struct _printf_t *p, char ch)
1188 {
1189  char *str;
1190  if (p->len )
1191  {
1192  if(ch)
1193  {
1194  p->len--;
1195  p->sent++;
1196  str = (char *) p->buffer;
1197  *str++ = ch;
1198  p->buffer = (void *) str;
1199  }
1200  }
1201  *((char *)p->buffer) = 0;
1202 }
1203 
1204 
1205 #ifdef PRINTF_TEST
1206 #ifdef DEFINE_PRINTF
1207 #error DEFINE_PRINTF must not be defined when testing
1208 #endif
1209 #endif
1210 
1211 #ifndef PRINTF_TEST
1212 // =============================================
1219 MEMSPACE
1220 int vsnprintf(char* str, size_t size, const char *format, va_list va)
1221 {
1222 
1223  int len;
1224  char *save = str;
1225  printf_t fn;
1226 
1227  *str = 0;
1228 
1229  fn.put = _putc_buffer_fn;
1230  fn.len = size;
1231  fn.sent = 0;
1232  fn.buffer = (void *) str;
1233 
1234  _printf_fn(&fn, format, va);
1235 
1236 // FIXME check size should == fn.size on exit
1237  len = strlen(save);
1238  return( len );
1239 }
1240 
1241 
1242 // =============================================
1249 MEMSPACE
1250 int snprintf(char* str, size_t size, const char *format, ...)
1251 {
1252  int len;
1253  va_list va;
1254 
1255  va_start(va, format);
1256  len = vsnprintf(str, size, format, va);
1257  va_end(va);
1258 
1259  return len;
1260 }
1261 
1262 
1263 #ifdef AVR
1264 MEMSPACE
1271 int vsnprintf_P(char* str, size_t size, __memx const char *format, va_list va)
1272 {
1273 
1274  int len;
1275  char *save = str;
1276  printf_t fn;
1277 
1278  *str = 0;
1279 
1280  fn.put = _putc_buffer_fn;
1281  fn.len = size;
1282  fn.sent = 0;
1283  fn.buffer = (void *) str;
1284 
1285  _printf_fn(&fn, format, va);
1286 
1287 // FIXME check size should == fn.size on exit
1288  len = strlen(save);
1289  return( len );
1290 }
1291 
1292 
1299 MEMSPACE
1300 int snprintf_P(char* str, size_t size, __memx const char *format, ...)
1301 {
1302  int len;
1303  va_list va;
1304 
1305  va_start(va, format);
1306  len = vsnprintf_P(str, size, format, va);
1307  va_end(va);
1308 
1309  return len;
1310 }
1311 
1312 
1318 MEMSPACE
1319 int sprintf_P(char* str, __memx const char *format, ...)
1320 {
1321  int len;
1322  va_list va;
1323 
1324  va_start(va, format);
1325 // FIXME max string size limit !!!!
1326  len = vsnprintf_P(str, 1024, format, va);
1327  va_end(va);
1328 
1329  return len;
1330 }
1331 #endif
1332 // =============================================
1334 // =============================================
1335 
1336 #ifdef DEFINE_PRINTF
1337 // =============================================
1343 MEMSPACE
1344 static void _putc_fn(struct _printf_t *p, char ch)
1345 {
1346  p->sent++;
1347  putchar(ch);
1348 }
1349 
1350 
1358 MEMSPACE
1359 int
1360 printf(const char *format, ...)
1361 {
1362  printf_t fn;
1363  va_list va;
1364 
1365  fn.put = _putc_fn;
1366  fn.sent = 0;
1367 
1368  va_start(va, format);
1369  _printf_fn(&fn, format, va);
1370  va_end(va);
1371 
1372  return ((int)fn.sent);
1373 }
1374 
1375 
1376 #ifdef AVR
1377 MEMSPACE
1385 int
1386 printf_P(__memx const char *format, ...)
1387 {
1388  printf_t fn;
1389  va_list va;
1390 
1391  fn.put = _putc_fn;
1392  fn.sent = 0;
1393 
1394  va_start(va, format);
1395  _printf_fn(&fn, format, va);
1396  va_end(va);
1397 
1398  return ((int)fn.sent);
1399 }
1400 #endif
1401 #endif
1402 #endif
_printf_t::buffer
void * buffer
Definition: mathio.h:83
printf
MEMSPACE int printf(const char *format,...)
p_ch_t::max
int max
current string index
Definition: mathio.h:111
_printf_t::put
void(* put)(struct _printf_t *, char)
Definition: mathio.h:82
MEMSPACE
#define MEMSPACE
Definition: user_config.h:17
p_ch_t
Data structure for character buffer with limits.
Definition: mathio.h:107
print_flags
MEMSPACE void print_flags(f_t f)
print flags set in t_t structure
Definition: printf.c:269
f_t::width
unsigned short width
Definition: mathio.h:93
putchar
int putchar(int c)
put a character to stdout See fdevopen() sets stream->put get for TTY devices
Definition: posix.c:363
_puts_pad
MEMSPACE void _puts_pad(printf_t *fn, char *s, int width, int count, int left)
put string, via user function, count bytes long, padded up to width, left or right aligned
Definition: printf.c:678
reverse
MEMSPACE void WEAK_ATR reverse(char *str)
Reverse a string in place Example: abcdef -> fedcba.
Definition: printf.c:90
f_t::neg
unsigned short neg
Definition: mathio.h:99
pch_init
MEMSPACE void pch_init(p_ch_t *p, char *str, int max)
Initialize character buffer with limits.
Definition: printf.c:218
f_t::space
unsigned short space
Definition: mathio.h:97
p_ftoa
MEMSPACE int p_ftoa(double val, char *str, int max, int width, int prec, f_t f)
p_ch_t::str
char * str
Definition: mathio.h:109
_printf_t
undefine any potential macro version of these functions
Definition: mathio.h:80
bin2num
MEMSPACE int bin2num(uint8_t *str, int strmax, int nummin, int base, uint8_t *nump, int numsize, int sign_ch)
Convert an unsigned number (numsize bytes in size) to ASCII in specified base Notes: No limit except ...
Definition: printf.c:143
strlen
MEMSPACE size_t WEAK_ATR strlen(const char *str)
String Length.
Definition: printf.c:53
_printf_t::sent
int sent
Definition: mathio.h:85
NULL
#define NULL
Definition: user_config.h:85
f_t::all
unsigned short all
Definition: mathio.h:102
isdigit
MEMSPACE int WEAK_ATR isdigit(int c)
test if a character is a digit
Definition: printf.c:71
pch
MEMSPACE int pch(p_ch_t *p, char ch)
Put character in buffer with limits.
Definition: printf.c:233
f_t::alt
unsigned short alt
Definition: mathio.h:100
pch_max_ind
MEMSPACE int pch_max_ind(p_ch_t *p)
Return maximum valid index for character buffer.
Definition: printf.c:256
vsnprintf
MEMSPACE int vsnprintf(char *str, size_t size, const char *format, va_list va)
vsnprintf function
Definition: printf.c:1220
p_etoa
MEMSPACE int p_etoa(double val, char *str, int max, int width, int prec, f_t f)
__memx
#define __memx
Definition: mathio.h:38
pch_ind
MEMSPACE int pch_ind(p_ch_t *p)
Return current index of character buffer with limits.
Definition: printf.c:247
p_ntoa
MEMSPACE int p_ntoa(uint8_t *nump, int numsize, char *str, int strmax, int radix, int width, int prec, f_t f)
Convert number an base 2 .. 16 to ASCII with optional sign Notes: 1) Numbers can be any number of dig...
Definition: printf.c:314
p_ch_t::ind
int ind
base of string to write to
Definition: mathio.h:110
f_t::prec
unsigned short prec
Definition: mathio.h:94
f_t::b
struct f_t::@0 b
f_t::plus
unsigned short plus
Definition: mathio.h:95
f_t::left
unsigned short left
Definition: mathio.h:96
f_t
format specifier flags
Definition: mathio.h:89
mathio.h
Math IO functions, and verious conversion code with floating point support.
_printf_t::len
int len
Definition: mathio.h:84
snprintf
MEMSPACE int snprintf(char *str, size_t size, const char *format,...)
snprintf function
Definition: printf.c:1250
_printf_fn
MEMSPACE void _printf_fn(printf_t *fn, __memx const char *fmt, va_list va)
vsnprintf function
Definition: printf.c:806
WEAK_ATR
#define WEAK_ATR
Definition: stringsup.h:33
_putc_buffer_fn
MEMSPACE void _putc_buffer_fn(struct _printf_t *p, char ch)
_putc_buffer_fn - character output to a string buffer Used by snprintf and vsnprintf You can make _pr...
Definition: printf.c:1187
f_t::zero
unsigned short zero
Definition: mathio.h:98
strupper
MEMSPACE void WEAK_ATR strupper(char *str)
UPPERCASE a string.
Definition: printf.c:112