|  | // Copyright 2016 The SwiftShader Authors. All Rights Reserved. | 
|  | // | 
|  | // Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | // you may not use this file except in compliance with the License. | 
|  | // You may obtain a copy of the License at | 
|  | // | 
|  | //    http://www.apache.org/licenses/LICENSE-2.0 | 
|  | // | 
|  | // Unless required by applicable law or agreed to in writing, software | 
|  | // distributed under the License is distributed on an "AS IS" BASIS, | 
|  | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | // See the License for the specific language governing permissions and | 
|  | // limitations under the License. | 
|  |  | 
|  | #ifndef _INFOSINK_INCLUDED_ | 
|  | #define _INFOSINK_INCLUDED_ | 
|  |  | 
|  | #include <math.h> | 
|  | #include "Common.h" | 
|  |  | 
|  | // Returns the fractional part of the given floating-point number. | 
|  | inline float fractionalPart(float f) { | 
|  | float intPart = 0.0f; | 
|  | return modff(f, &intPart); | 
|  | } | 
|  |  | 
|  | // | 
|  | // TPrefixType is used to centralize how info log messages start. | 
|  | // See below. | 
|  | // | 
|  | enum TPrefixType { | 
|  | EPrefixNone, | 
|  | EPrefixInfo, | 
|  | EPrefixWarning, | 
|  | EPrefixError, | 
|  | EPrefixInternalError, | 
|  | EPrefixUnimplemented, | 
|  | EPrefixNote | 
|  | }; | 
|  |  | 
|  | // | 
|  | // Encapsulate info logs for all objects that have them. | 
|  | // | 
|  | // The methods are a general set of tools for getting a variety of | 
|  | // messages and types inserted into the log. | 
|  | // | 
|  | class TInfoSinkBase { | 
|  | public: | 
|  | TInfoSinkBase() {} | 
|  |  | 
|  | template <typename T> | 
|  | TInfoSinkBase& operator<<(const T& t) { | 
|  | TPersistStringStream stream; | 
|  | stream << t; | 
|  | sink.append(stream.str()); | 
|  | return *this; | 
|  | } | 
|  | // Override << operator for specific types. It is faster to append strings | 
|  | // and characters directly to the sink. | 
|  | TInfoSinkBase& operator<<(char c) { | 
|  | sink.append(1, c); | 
|  | return *this; | 
|  | } | 
|  | TInfoSinkBase& operator<<(const char* str) { | 
|  | sink.append(str); | 
|  | return *this; | 
|  | } | 
|  | TInfoSinkBase& operator<<(const TPersistString& str) { | 
|  | sink.append(str); | 
|  | return *this; | 
|  | } | 
|  | TInfoSinkBase& operator<<(const TString& str) { | 
|  | sink.append(str.c_str()); | 
|  | return *this; | 
|  | } | 
|  | // Make sure floats are written with correct precision. | 
|  | TInfoSinkBase& operator<<(float f) { | 
|  | // Make sure that at least one decimal point is written. If a number | 
|  | // does not have a fractional part, the default precision format does | 
|  | // not write the decimal portion which gets interpreted as integer by | 
|  | // the compiler. | 
|  | TPersistStringStream stream; | 
|  | if (fractionalPart(f) == 0.0f) { | 
|  | stream.precision(1); | 
|  | stream << std::showpoint << std::fixed << f; | 
|  | } else { | 
|  | stream.unsetf(std::ios::fixed); | 
|  | stream.unsetf(std::ios::scientific); | 
|  | stream.precision(8); | 
|  | stream << f; | 
|  | } | 
|  | sink.append(stream.str()); | 
|  | return *this; | 
|  | } | 
|  | // Write boolean values as their names instead of integral value. | 
|  | TInfoSinkBase& operator<<(bool b) { | 
|  | const char* str = b ? "true" : "false"; | 
|  | sink.append(str); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | void erase() { sink.clear(); } | 
|  | int size() { return static_cast<int>(sink.size()); } | 
|  |  | 
|  | const TPersistString& str() const { return sink; } | 
|  | const char* c_str() const { return sink.c_str(); } | 
|  |  | 
|  | void prefix(TPrefixType message); | 
|  | void location(const TSourceLoc& loc); | 
|  | void message(TPrefixType message, const char* s); | 
|  | void message(TPrefixType message, const char* s, TSourceLoc loc); | 
|  |  | 
|  | private: | 
|  | TPersistString sink; | 
|  | }; | 
|  |  | 
|  | class TInfoSink { | 
|  | public: | 
|  | TInfoSinkBase info; | 
|  | TInfoSinkBase debug; | 
|  | TInfoSinkBase obj; | 
|  | }; | 
|  |  | 
|  | #endif // _INFOSINK_INCLUDED_ |