| // 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, |
| 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_ |