192 lines
4.9 KiB
C++
192 lines
4.9 KiB
C++
/*
|
|
* Copyright (c) 2002-2006 The Regents of The University of Michigan
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are
|
|
* met: redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer;
|
|
* redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution;
|
|
* neither the name of the copyright holders nor the names of its
|
|
* contributors may be used to endorse or promote products derived from
|
|
* this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
* Authors: Nathan Binkert
|
|
* Steve Reinhardt
|
|
*/
|
|
|
|
#ifndef __BASE_CPRINTF_HH__
|
|
#define __BASE_CPRINTF_HH__
|
|
|
|
#include <ios>
|
|
#include <iostream>
|
|
#include <list>
|
|
#include <string>
|
|
|
|
#include "base/varargs.hh"
|
|
#include "base/cprintf_formats.hh"
|
|
|
|
namespace cp {
|
|
|
|
#define CPRINTF_DECLARATION VARARGS_DECLARATION(cp::Print)
|
|
#define CPRINTF_DEFINITION VARARGS_DEFINITION(cp::Print)
|
|
|
|
struct Print
|
|
{
|
|
protected:
|
|
std::ostream &stream;
|
|
const char *format;
|
|
const char *ptr;
|
|
bool cont;
|
|
|
|
std::ios::fmtflags saved_flags;
|
|
char saved_fill;
|
|
int saved_precision;
|
|
|
|
Format fmt;
|
|
void process();
|
|
|
|
public:
|
|
Print(std::ostream &stream, const std::string &format);
|
|
Print(std::ostream &stream, const char *format);
|
|
~Print();
|
|
|
|
int
|
|
get_number(int data)
|
|
{
|
|
return data;
|
|
}
|
|
|
|
template <typename T>
|
|
int
|
|
get_number(const T& data)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
template <typename T>
|
|
void
|
|
add_arg(const T &data)
|
|
{
|
|
if (!cont)
|
|
process();
|
|
|
|
if (fmt.get_width) {
|
|
fmt.get_width = false;
|
|
cont = true;
|
|
fmt.width = get_number(data);
|
|
return;
|
|
}
|
|
|
|
if (fmt.get_precision) {
|
|
fmt.get_precision = false;
|
|
cont = true;
|
|
fmt.precision = get_number(data);
|
|
return;
|
|
}
|
|
|
|
switch (fmt.format) {
|
|
case Format::character:
|
|
format_char(stream, data, fmt);
|
|
break;
|
|
|
|
case Format::integer:
|
|
format_integer(stream, data, fmt);
|
|
break;
|
|
|
|
case Format::floating:
|
|
format_float(stream, data, fmt);
|
|
break;
|
|
|
|
case Format::string:
|
|
format_string(stream, data, fmt);
|
|
break;
|
|
|
|
default:
|
|
stream << "<bad format>";
|
|
break;
|
|
}
|
|
}
|
|
|
|
void end_args();
|
|
};
|
|
|
|
/* end namespace cp */ }
|
|
|
|
typedef VarArgs::List<cp::Print> CPrintfArgsList;
|
|
|
|
inline void
|
|
ccprintf(std::ostream &stream, const char *format, const CPrintfArgsList &args)
|
|
{
|
|
cp::Print print(stream, format);
|
|
args.add_args(print);
|
|
}
|
|
|
|
inline void
|
|
ccprintf(std::ostream &stream, const char *format, CPRINTF_DECLARATION)
|
|
{
|
|
cp::Print print(stream, format);
|
|
VARARGS_ADDARGS(print);
|
|
}
|
|
|
|
inline void
|
|
cprintf(const char *format, CPRINTF_DECLARATION)
|
|
{
|
|
ccprintf(std::cout, format, VARARGS_ALLARGS);
|
|
}
|
|
|
|
inline std::string
|
|
csprintf(const char *format, CPRINTF_DECLARATION)
|
|
{
|
|
std::stringstream stream;
|
|
ccprintf(stream, format, VARARGS_ALLARGS);
|
|
return stream.str();
|
|
}
|
|
|
|
/*
|
|
* functions again with std::string. We have both so we don't waste
|
|
* time converting const char * to std::string since we don't take
|
|
* advantage of it.
|
|
*/
|
|
inline void
|
|
ccprintf(std::ostream &stream, const std::string &format,
|
|
const CPrintfArgsList &args)
|
|
{
|
|
ccprintf(stream, format.c_str(), args);
|
|
}
|
|
|
|
inline void
|
|
ccprintf(std::ostream &stream, const std::string &format, CPRINTF_DECLARATION)
|
|
{
|
|
ccprintf(stream, format.c_str(), VARARGS_ALLARGS);
|
|
}
|
|
|
|
inline void
|
|
cprintf(const std::string &format, CPRINTF_DECLARATION)
|
|
{
|
|
ccprintf(std::cout, format.c_str(), VARARGS_ALLARGS);
|
|
}
|
|
|
|
inline std::string
|
|
csprintf(const std::string &format, CPRINTF_DECLARATION)
|
|
{
|
|
std::stringstream stream;
|
|
ccprintf(stream, format.c_str(), VARARGS_ALLARGS);
|
|
return stream.str();
|
|
}
|
|
|
|
#endif // __CPRINTF_HH__
|