mirror of
				https://github.com/pascallanger/DIY-Multiprotocol-TX-Module.git
				synced 2025-10-30 18:55:21 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			382 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			382 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  Print.cpp - Base class that provides print() and println()
 | |
|  Copyright (c) 2008 David A. Mellis.  All right reserved.
 | |
| 
 | |
|  This library is free software; you can redistribute it and/or
 | |
|  modify it under the terms of the GNU Lesser General Public
 | |
|  License as published by the Free Software Foundation; either
 | |
|  version 2.1 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
 | |
|  Lesser General Public License for more details.
 | |
| 
 | |
|  You should have received a copy of the GNU Lesser General Public
 | |
|  License along with this library; if not, write to the Free Software
 | |
|  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
 | |
| 
 | |
|  Modified 23 November 2006 by David A. Mellis
 | |
|  */
 | |
| 
 | |
| #include <stdlib.h>
 | |
| #include <stdio.h>
 | |
| #include <string.h>
 | |
| #include <math.h>
 | |
| #include "Arduino.h"
 | |
| 
 | |
| #include "Print.h"
 | |
| 
 | |
| // Public Methods //////////////////////////////////////////////////////////////
 | |
| 
 | |
| /* default implementation: may be overridden */
 | |
| size_t Print::write(const uint8_t *buffer, size_t size)
 | |
| {
 | |
|   size_t n = 0;
 | |
|   while (size--)
 | |
|   {
 | |
|     register size_t cb = write(*buffer++);
 | |
| 
 | |
|     if(!cb) // error return (prevents infinite loops)
 | |
|     {
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     n += cb;
 | |
|   }
 | |
|   return n;
 | |
| }
 | |
| 
 | |
| size_t Print::print(const __FlashStringHelper *ifsh)
 | |
| {
 | |
| PGM_P p = reinterpret_cast<PGM_P>(ifsh);
 | |
| size_t n = 0;
 | |
| char tbuf[32]; // will write 32 chars at a time (helps for USB)
 | |
| register char *p1;
 | |
| 
 | |
| 
 | |
|   p1 = tbuf;
 | |
| 
 | |
|   while (1)
 | |
|   {
 | |
|     unsigned char c = pgm_read_byte(p++);
 | |
| 
 | |
|     if(c == 0)
 | |
|     {
 | |
|       if(p1 > &(tbuf[0]))
 | |
|       {
 | |
|         n += write(tbuf, p1 - &(tbuf[0])); // write whatever I've got
 | |
|       }
 | |
| 
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     if(p1 >= &(tbuf[sizeof(tbuf)]))
 | |
|     {
 | |
|       register size_t cb = write(tbuf, p1 - &(tbuf[0]));
 | |
| 
 | |
|       if(!cb) // error (prevents infinite loops)
 | |
|       {
 | |
|         break;
 | |
|       }
 | |
| 
 | |
|       n += cb;
 | |
|       p1 = tbuf;
 | |
|     }
 | |
| 
 | |
|     *(p1++) = c;
 | |
|   }
 | |
| 
 | |
|   return n;
 | |
| }
 | |
| 
 | |
| size_t Print::print(const String &s)
 | |
| {
 | |
|   return write(s.c_str(), s.length());
 | |
| }
 | |
| 
 | |
| size_t Print::print(const char str[])
 | |
| {
 | |
|   return write(str);
 | |
| }
 | |
| 
 | |
| size_t Print::print(char c)
 | |
| {
 | |
|   return write(c);
 | |
| }
 | |
| 
 | |
| size_t Print::print(unsigned char b, int base)
 | |
| {
 | |
|   return print((unsigned long) b, base);
 | |
| }
 | |
| 
 | |
| size_t Print::print(int n, int base)
 | |
| {
 | |
|   return print((long) n, base);
 | |
| }
 | |
| 
 | |
| size_t Print::print(unsigned int n, int base)
 | |
| {
 | |
|   return print((unsigned long) n, base);
 | |
| }
 | |
| 
 | |
| size_t Print::print(long n, int base)
 | |
| {
 | |
|   if (base == 0)
 | |
|   {
 | |
|     return write(n);
 | |
|   }
 | |
|   else if (base == 10)
 | |
|   {
 | |
|     if (n < 0)
 | |
|     {
 | |
|       int t = print('-');
 | |
|       n = -n;
 | |
|       return printNumber(n, 10) + t;
 | |
|     }
 | |
|     return printNumber(n, 10);
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     return printNumber(n, base);
 | |
|   }
 | |
| }
 | |
| 
 | |
| size_t Print::print(unsigned long n, int base)
 | |
| {
 | |
|   if (base == 0)
 | |
|     return write(n);
 | |
|   else
 | |
|     return printNumber(n, base);
 | |
| }
 | |
| 
 | |
| size_t Print::print(double n, int digits)
 | |
| {
 | |
|   return printFloat(n, digits);
 | |
| }
 | |
| 
 | |
| size_t Print::println(const __FlashStringHelper *ifsh)
 | |
| {
 | |
|   size_t n = print(ifsh);
 | |
|   n += println();
 | |
|   return n;
 | |
| }
 | |
| 
 | |
| size_t Print::print(const Printable& x)
 | |
| {
 | |
|   return x.printTo(*this);
 | |
| }
 | |
| 
 | |
| size_t Print::println(void)
 | |
| {
 | |
| char tbuf[2];
 | |
| 
 | |
|   tbuf[0] = '\r';  // this should be more efficient, and smaller
 | |
|   tbuf[1] = '\n';
 | |
|   return write(tbuf, 2);
 | |
| //  size_t n = print('\r');
 | |
| //  n += print('\n');
 | |
| //  return n;
 | |
| }
 | |
| 
 | |
| size_t Print::println(const String &s)
 | |
| {
 | |
|   size_t n = print(s);
 | |
|   n += println();
 | |
|   return n;
 | |
| }
 | |
| 
 | |
| size_t Print::println(const char c[])
 | |
| {
 | |
|   size_t n = print(c);
 | |
|   n += println();
 | |
|   return n;
 | |
| }
 | |
| 
 | |
| size_t Print::println(char c)
 | |
| {
 | |
|   size_t n = print(c);
 | |
|   n += println();
 | |
|   return n;
 | |
| }
 | |
| 
 | |
| size_t Print::println(unsigned char b, int base)
 | |
| {
 | |
|   size_t n = print(b, base);
 | |
|   n += println();
 | |
|   return n;
 | |
| }
 | |
| 
 | |
| size_t Print::println(int num, int base)
 | |
| {
 | |
|   size_t n = print(num, base);
 | |
|   n += println();
 | |
|   return n;
 | |
| }
 | |
| 
 | |
| size_t Print::println(unsigned int num, int base)
 | |
| {
 | |
|   size_t n = print(num, base);
 | |
|   n += println();
 | |
|   return n;
 | |
| }
 | |
| 
 | |
| size_t Print::println(long num, int base)
 | |
| {
 | |
|   size_t n = print(num, base);
 | |
|   n += println();
 | |
|   return n;
 | |
| }
 | |
| 
 | |
| size_t Print::println(unsigned long num, int base)
 | |
| {
 | |
|   size_t n = print(num, base);
 | |
|   n += println();
 | |
|   return n;
 | |
| }
 | |
| 
 | |
| size_t Print::println(double num, int digits)
 | |
| {
 | |
|   size_t n = print(num, digits);
 | |
|   n += println();
 | |
|   return n;
 | |
| }
 | |
| 
 | |
| size_t Print::println(const Printable& x)
 | |
| {
 | |
|   size_t n = print(x);
 | |
|   n += println();
 | |
|   return n;
 | |
| }
 | |
| 
 | |
| // Private Methods /////////////////////////////////////////////////////////////
 | |
| 
 | |
| size_t Print::printNumber(unsigned long n, uint8_t base)
 | |
| {
 | |
|   char buf[8 * sizeof(long) + 1]; // Assumes 8-bit chars plus zero byte.
 | |
|   char *str = &buf[sizeof(buf) - 1];
 | |
| 
 | |
|   *str = '\0';
 | |
| 
 | |
|   // prevent crash if called with base == 1
 | |
|   if (base < 2)
 | |
|     base = 10;
 | |
| 
 | |
|   do
 | |
|   {
 | |
|     unsigned long m = n;
 | |
|     n /= base;
 | |
|     char c = m - base * n;
 | |
|     *--str = c < 10 ? c + '0' : c + 'A' - 10;
 | |
|   } while(n);
 | |
| 
 | |
|   return write(str);
 | |
| }
 | |
| 
 | |
| size_t Print::printFloat(double number, uint8_t digits)
 | |
| {
 | |
|   size_t n = 0;
 | |
| 
 | |
|   if (isnan(number))
 | |
|     return print("nan");
 | |
|   if (isinf(number))
 | |
|     return print("inf");
 | |
|   if (number > 4294967040.0)
 | |
|     return print ("ovf");  // constant determined empirically
 | |
|   if (number <-4294967040.0)
 | |
|     return print ("ovf");  // constant determined empirically
 | |
| 
 | |
|   // Handle negative numbers
 | |
|   if (number < 0.0)
 | |
|   {
 | |
|      n += print('-');
 | |
|      number = -number;
 | |
|   }
 | |
| 
 | |
|   // Round correctly so that print(1.999, 2) prints as "2.00"
 | |
|   double rounding = 0.5;
 | |
|   for (uint8_t i=0; i<digits; ++i)
 | |
|   {
 | |
|     rounding /= 10.0;
 | |
|   }
 | |
| 
 | |
|   number += rounding;
 | |
| 
 | |
|   // Extract the integer part of the number and print it
 | |
|   unsigned long int_part = (unsigned long)number;
 | |
|   double remainder = number - (double)int_part;
 | |
|   n += print(int_part);
 | |
| 
 | |
|   // Print the decimal point, but only if there are digits beyond
 | |
|   if (digits > 0)
 | |
|   {
 | |
|     n += print(".");
 | |
|   }
 | |
| 
 | |
|   // Extract digits from the remainder one at a time
 | |
|   while (digits-- > 0)
 | |
|   {
 | |
|     remainder *= 10.0;
 | |
|     int toPrint = int(remainder);
 | |
|     n += print(toPrint);
 | |
|     remainder -= toPrint;
 | |
|   }
 | |
| 
 | |
|   return n;
 | |
| }
 | |
| 
 | |
| // things that I added
 | |
| 
 | |
| int Print::printf(const char *pszFormat, ...) /*__attribute__ ((format(printf, 2, 3)))*/ // added API for 'printf' since it has been suggested...
 | |
| {
 | |
| va_list va;
 | |
| int cbOut;
 | |
| 
 | |
|   va_start(va, pszFormat);
 | |
| 
 | |
|   cbOut = vsnprintf(NULL, 0, pszFormat, va);
 | |
| 
 | |
|   if(cbOut > 0)
 | |
|   {
 | |
|     char *p1 = (char *)malloc(cbOut + 2);
 | |
| 
 | |
|     if(p1)
 | |
|     {
 | |
|       cbOut = vsnprintf(p1, cbOut + 1, pszFormat, va);
 | |
|       print(p1);
 | |
|       free(p1);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   va_end(va);
 | |
| 
 | |
|   return cbOut;
 | |
| }
 | |
| 
 | |
| int Print::printf_P(const char *pszFormat, ...) /*__attribute__ ((format(printf_P, 2, 3)))*/ // added API for 'printf' since it has been suggested...
 | |
| {
 | |
| va_list va;
 | |
| int cbOut;
 | |
| 
 | |
|   va_start(va, pszFormat);
 | |
| 
 | |
|   cbOut = vsnprintf_P(NULL, 0, (const char *)pszFormat, va);
 | |
| 
 | |
|   if(cbOut > 0)
 | |
|   {
 | |
|     char *p1 = (char *)malloc(cbOut + 2);
 | |
| 
 | |
|     if(p1)
 | |
|     {
 | |
|       cbOut = vsnprintf_P(p1, cbOut + 1, (const char *)pszFormat, va);
 | |
|       print(p1);
 | |
|       free(p1);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   va_end(va);
 | |
| 
 | |
|   return cbOut;
 | |
| }
 | |
| 
 |