#ifndef UTIL_H
#define UTIL_H

#include <cstring>
#include <memory>
#include <string>
#include <windows.h>

#define CONCAT_IMPL(a, b) a##b
#define CONCAT(a, b) CONCAT_IMPL(a, b)
#define _ CONCAT(unused_, __LINE__)

template <typename F>
struct Finally
{
	F f;
	inline Finally(F f) : f(f) {}
	inline ~Finally() { f(); }
};
#define FINALLY Finally _ = [=]()

/* Generic RAII type. */
template <typename T, auto F, typename U, auto E = 0>
struct Managed
{
	T obj;
	Managed(T obj) : obj(obj)
	{
		if (obj == reinterpret_cast<T>(E))
			throw U();
	}
	~Managed() { F(obj); }
	operator T() { return obj; }
	auto& operator *() { return *obj; }
	auto& operator ->() { return obj; }
	auto& operator [](size_t i) { return obj[i]; }
};

/* Buf is a span-like structure of a buffer and its size. */
template <typename T>
struct Buf
{
	T* data;
	size_t c;
	Buf(T* data, size_t c) : data(data), c(c) {}
	Buf(std::basic_string<T>& s) : data(s.data()), c(s.capacity()) {}
	template <size_t N> Buf(T (&data)[N]) : data(data), c(N) {}
	operator T*() { return data; }
	T& operator *() { return *data; }
	T& operator [](size_t i) { return data[i]; }
	Buf<T> operator +(size_t i) { return {data+i, c-i}; }
	//T operator -(size_t i) { return {data-i, c+i}; }
};

inline int Cmp(const int a, const int b)
{
	if (a == b) return 0;
	if (a > b) return 1;
	return -1;
}

inline size_t Min(size_t a, size_t b)
{
	return a < b? a: b;
}

template <typename T, size_t N>
inline size_t Len(T (&)[N])
{
	return N-1;
}

/* Format wide string. */
template<typename... T>
inline int Swprintf(Buf<wchar_t> buf, const wchar_t* const fmt, T... xs)
{
	return _snwprintf_s(buf, buf.c, _TRUNCATE, fmt, xs...);
}

/* Format static narrow string. */
template<typename... T>
inline int Sprintf(Buf<char> buf, const char* const fmt, T... xs)
{
	return _snprintf_s(buf, buf.c, _TRUNCATE, fmt, xs...);
}

/* Copy to static wide string buffer. */
inline wchar_t* Wcscpy(Buf<wchar_t> dst, const wchar_t* const src)
{
	const size_t len = Min(dst.c, wcslen(src)+1);
	memcpy(dst, src, len*sizeof(wchar_t));
	dst[len-1] = 0;
	return dst;
}

/* Copy to static narrow string buffer. */
inline char* Strcpy(Buf<char> dst, const char* const src)
{
	const size_t len = Min(dst.c, strlen(src)+1);
	memcpy(dst, src, len);
	dst[len-1] = 0;
	return dst;
}

#endif