/*
 * Copyright (c) 2011 Remko Tronçon
 * Licensed under the GNU General Public License v3.
 * See Documentation/Licenses/GPLv3.txt for more information.
 */

#include <iostream>
#include <set>
#include <vector>
#include <cassert>
#include <boost/algorithm/string/predicate.hpp>
#include <boost/graph/graph_traits.hpp>
#include <boost/graph/adjacency_list.hpp>
#include <boost/graph/topological_sort.hpp>
#include <boost/graph/topological_sort.hpp>
#include <boost/graph/graphviz.hpp>

// -----------------------------------------------------------------------------
// Include diagnostics data from CLang
// -----------------------------------------------------------------------------

#define DIAG(name, a, b, c, d, e, f, g) name,
	
namespace diag {
	enum LexKinds {
#include <clang/Basic/DiagnosticLexKinds.inc>
#include <clang/Basic/DiagnosticParseKinds.inc>
#include <clang/Basic/DiagnosticCommonKinds.inc>
#include <clang/Basic/DiagnosticDriverKinds.inc>
#include <clang/Basic/DiagnosticFrontendKinds.inc>
#include <clang/Basic/DiagnosticSemaKinds.inc>
	};
}

#define GET_DIAG_ARRAYS
#include <clang/Basic/DiagnosticGroups.inc>
#undef GET_DIAG_ARRAYS

struct DiagTableEntry {
	const char* name;
	const short* array;
	const short* group;
};

static const DiagTableEntry diagnostics[] = {
#define GET_DIAG_TABLE
#include <clang/Basic/DiagnosticGroups.inc>
#undef GET_DIAG_TABLE
};
static const size_t diagnostics_count = sizeof(diagnostics) / sizeof(diagnostics[0]);

// -----------------------------------------------------------------------------

using namespace boost;

struct Properties {
	Properties() : have(false), implicitHave(false), dontWant(false), implicitDontWant(false), ignored(false), available(false), missing(false), redundant(false), alreadyCovered(false) {
	}

	std::string name;
	bool have;
	bool implicitHave;
	bool dontWant;
	bool implicitDontWant;
	bool ignored;
	bool available;
	bool missing;
	bool redundant;
	bool alreadyCovered;
};

class GraphVizLabelWriter {
	public:
		GraphVizLabelWriter(const std::vector<Properties>& properties) : properties(properties) {
		}	
		
		template <class VertexOrEdge>
		void operator()(std::ostream& out, const VertexOrEdge& v) const {
			std::string color;
			if (properties[v].missing) {
				color = "orange";
			}
			else if (properties[v].redundant) {
				color = "lightblue";
			}
			else if (properties[v].have) {
				color = "darkgreen";
			}
			else if (properties[v].implicitHave) {
				color = "green";
			}
			else if (properties[v].dontWant) {
				color = "red";
			}
			else if (properties[v].implicitDontWant) {
				color = "pink";
			}
			else if (properties[v].ignored) {
				color = "white";
			}
			else if (properties[v].available) {
				color = "yellow";
			}
			else {
				assert(false);
			}
			out << "[label=" << escape_dot_string(properties[v].name) << " fillcolor=\"" << color << "\" style=filled]";
		}

	private:
		const std::vector<Properties> properties;
};

int main(int argc, char* argv[]) {
	// Parse command-line arguments
	std::set<std::string> have;
	std::set<std::string> dontWant;
	std::string outputDir;
	for (int i = 1; i < argc; ++i) {
		std::string arg(argv[i]);
		if (starts_with(arg, "-W")) {
			have.insert(arg.substr(2, arg.npos));
		}
		else if (starts_with(arg, "-w")) {
			dontWant.insert(arg.substr(2, arg.npos));
		}
		else if (starts_with(arg, "-O")) {
			outputDir = arg.substr(2, arg.npos) + "/";
		}
	}

	// Build the graph and initialize properties
	typedef adjacency_list<vecS, vecS, bidirectionalS> Graph;
	typedef graph_traits<Graph>::vertex_descriptor Vertex;
	Graph g(diagnostics_count);
	std::vector<Properties> properties(num_vertices(g));
	for (size_t i = 0; i < diagnostics_count; ++i) {
		std::string name(diagnostics[i].name);
		properties[i].name = name;
		properties[i].implicitHave = properties[i].have = have.find(name) != have.end();
		properties[i].implicitDontWant = properties[i].dontWant = dontWant.find(name) != dontWant.end();
		properties[i].ignored = diagnostics[i].group == 0 && diagnostics[i].array == 0;
		properties[i].alreadyCovered = false;
		properties[i].available = true;
		for (const short* j = diagnostics[i].group; j && *j != -1; ++j) {
			add_edge(i, *j, g);
		}
	}

	// Sort the diagnostics
	std::list<Vertex> sortedDiagnostics;
	boost::topological_sort(g, std::front_inserter(sortedDiagnostics));

	// Propagate dontWant and have properties down
	for(std::list<Vertex>::const_iterator i = sortedDiagnostics.begin(); i != sortedDiagnostics.end(); ++i) {
		graph_traits<Graph>::adjacency_iterator adjacentIt, adjacentEnd;
		for (tie(adjacentIt, adjacentEnd) = adjacent_vertices(*i, g); adjacentIt != adjacentEnd; ++adjacentIt) {
			properties[*adjacentIt].implicitDontWant = properties[*i].implicitDontWant || properties[*adjacentIt].implicitDontWant;
			properties[*adjacentIt].implicitHave = properties[*i].implicitHave || properties[*adjacentIt].implicitHave;
		}
	}

	// Propagate 'available' property upwards
	for(std::list<Vertex>::const_reverse_iterator i = sortedDiagnostics.rbegin(); i != sortedDiagnostics.rend(); ++i) {
		properties[*i].available = properties[*i].available && !properties[*i].implicitDontWant;
		graph_traits<Graph>::in_edge_iterator edgesIt, edgesEnd;
		graph_traits<Graph>::edge_descriptor edge;
		for (tie(edgesIt, edgesEnd) = in_edges(*i, g); edgesIt != edgesEnd; ++edgesIt) {
			properties[source(*edgesIt, g)].available = properties[source(*edgesIt, g)].available && properties[*i].available;
		}
	}

	// Collect missing & redundant flags
	std::set<std::string> missing;
	std::set<std::string> redundant;
	for(std::list<Vertex>::const_iterator i = sortedDiagnostics.begin(); i != sortedDiagnostics.end(); ++i) {
		bool markChildrenCovered = true;
		if (properties[*i].alreadyCovered) {
			if (properties[*i].have) {
				properties[*i].redundant = true;
				redundant.insert(properties[*i].name);
			}
		}
		else {
			if (properties[*i].available) {
				if (!properties[*i].implicitHave && !properties[*i].ignored) {
					properties[*i].missing = true;
					missing.insert(properties[*i].name);
				}
			}
			else {
				markChildrenCovered = false;
			}
		}
		if (markChildrenCovered) {
			graph_traits<Graph>::adjacency_iterator adjacentIt, adjacentEnd;
			for (tie(adjacentIt, adjacentEnd) = adjacent_vertices(*i, g); adjacentIt != adjacentEnd; ++adjacentIt) {
				properties[*adjacentIt].alreadyCovered = true;
			}
		}
	}

	// Write information
	if (!missing.empty()) {
		std::cout << "Missing diagnostic flags: ";
		for(std::set<std::string>::const_iterator i = missing.begin(); i != missing.end(); ++i) {
			std::cout << "-W" << *i << " ";
		}
		std::cout<< std::endl;
	}

	if (!redundant.empty()) {
		std::cout << "Redundant diagnostic flags: ";
		for(std::set<std::string>::const_iterator i = redundant.begin(); i != redundant.end(); ++i) {
			std::cout << "-W" << *i << " ";
		}
		std::cout<< std::endl;
	}

	// Write graphviz file
	if (!outputDir.empty()) {
		std::ofstream f((outputDir + "clang-diagnostics-overview.dot").c_str());
		write_graphviz(f, g, GraphVizLabelWriter(properties));
		f.close();
	}

	return 0;
}