|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176 |
- /* Arduino SdFat Library
- * Copyright (C) 2012 by William Greiman
- *
- * This file is part of the Arduino SdFat Library
- *
- * This Library is free software: you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation, either version 3 of the License, or
- * (at your option) any later version.
- *
- * This Library is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with the Arduino SdFat Library. If not, see
- * <http://www.gnu.org/licenses/>.
- */
- #include <ostream.h>
- #ifndef PSTR
- #define PSTR(x) x
- #endif
- //------------------------------------------------------------------------------
- void ostream::do_fill(unsigned len) {
- for (; len < width(); len++) putch(fill());
- width(0);
- }
- //------------------------------------------------------------------------------
- void ostream::fill_not_left(unsigned len) {
- if ((flags() & adjustfield) != left) {
- do_fill(len);
- }
- }
- //------------------------------------------------------------------------------
- char* ostream::fmtNum(uint32_t n, char *ptr, uint8_t base) {
- char a = flags() & uppercase ? 'A' - 10 : 'a' - 10;
- do {
- uint32_t m = n;
- n /= base;
- char c = m - base * n;
- *--ptr = c < 10 ? c + '0' : c + a;
- } while (n);
- return ptr;
- }
- //------------------------------------------------------------------------------
- void ostream::putBool(bool b) {
- if (flags() & boolalpha) {
- if (b) {
- putPgm(PSTR("true"));
- } else {
- putPgm(PSTR("false"));
- }
- } else {
- putChar(b ? '1' : '0');
- }
- }
- //------------------------------------------------------------------------------
- void ostream::putChar(char c) {
- fill_not_left(1);
- putch(c);
- do_fill(1);
- }
- //------------------------------------------------------------------------------
- void ostream::putDouble(double n) {
- uint8_t nd = precision();
- double round = 0.5;
- char sign;
- char buf[13]; // room for sign, 10 digits, '.', and zero byte
- char *end = buf + sizeof(buf) - 1;
- char *str = end;
- // terminate string
- *end = '\0';
-
- // get sign and make nonnegative
- if (n < 0.0) {
- sign = '-';
- n = -n;
- } else {
- sign = flags() & showpos ? '+' : '\0';
- }
- // check for larger than uint32_t
- if (n > 4.0E9) {
- putPgm(PSTR("BIG FLT"));
- return;
- }
- // round up and separate in and fraction parts
- for (uint8_t i = 0; i < nd; ++i) round *= 0.1;
- n += round;
- uint32_t intPart = n;
- double fractionPart = n - intPart;
-
- // format intPart and decimal point
- if (nd || (flags() & showpoint)) *--str = '.';
- str = fmtNum(intPart, str, 10);
-
- // calculate length for fill
- uint8_t len = sign ? 1 : 0;
- len += nd + end - str;
-
- // extract adjust field
- fmtflags adj = flags() & adjustfield;
- if (adj == internal) {
- if (sign) putch(sign);
- do_fill(len);
- } else {
- // do fill for internal or right
- fill_not_left(len);
- if (sign) *--str = sign;
- }
- putstr(str);
- // output fraction
- while (nd-- > 0) {
- fractionPart *= 10.0;
- int digit = static_cast<int>(fractionPart);
- putch(digit + '0');
- fractionPart -= digit;
- }
- // do fill if not done above
- do_fill(len);
- }
- //------------------------------------------------------------------------------
- void ostream::putNum(int32_t n) {
- bool neg = n < 0 && flagsToBase() == 10;
- if (neg) n = -n;
- putNum(n, neg);
- }
- //------------------------------------------------------------------------------
- void ostream::putNum(uint32_t n, bool neg) {
- char buf[13];
- char* end = buf + sizeof(buf) - 1;
- char* num;
- char* str;
- uint8_t base = flagsToBase();
- *end = '\0';
- str = num = fmtNum(n, end, base);
- if (base == 10) {
- if (neg) {
- *--str = '-';
- } else if (flags() & showpos) {
- *--str = '+';
- }
- } else if (flags() & showbase) {
- if (flags() & hex) {
- *--str = flags() & uppercase ? 'X' : 'x';
- }
- *--str = '0';
- }
- uint8_t len = end - str;
- fmtflags adj = flags() & adjustfield;
- if (adj == internal) {
- while (str < num) putch(*str++);
- }
- if (adj != left) {
- do_fill(len);
- }
- putstr(str);
- do_fill(len);
- }
- //------------------------------------------------------------------------------
- void ostream::putPgm(const char* str) {
- int n;
- for (n = 0; pgm_read_byte(&str[n]); n++) {}
- fill_not_left(n);
- for (uint8_t c; (c = pgm_read_byte(str)); str++) {
- putch(c);
- }
- do_fill(n);
- }
- //------------------------------------------------------------------------------
- void ostream::putStr(const char *str) {
- unsigned n = strlen(str);
- fill_not_left(n);
- putstr(str);
- do_fill(n);
- }
|