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