/*
  This file is part of Ingen.
  Copyright 2007-2012 David Robillard <http://drobilla.net/>

  Ingen is free software: you can redistribute it and/or modify it under the
  terms of the GNU Affero General Public License as published by the Free
  Software Foundation, either version 3 of the License, or any later version.

  Ingen is distributed in the hope that it will be useful, but WITHOUT ANY
  WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
  A PARTICULAR PURPOSE.  See the GNU Affero General Public License for details.

  You should have received a copy of the GNU Affero General Public License
  along with Ingen.  If not, see <http://www.gnu.org/licenses/>.
*/

#ifndef INGEN_ENGINE_UTIL_HPP
#define INGEN_ENGINE_UTIL_HPP

#include <cstdlib>
#include <string>

#include "raul/log.hpp"
#include "raul/Path.hpp"

#include "ingen_config.h"

#include <fenv.h>
#ifdef __SSE__
#include <xmmintrin.h>
#endif

#ifdef __clang__
#    define REALTIME __attribute__((annotate("realtime")))
#else
#    define REALTIME
#endif

#ifdef USE_ASSEMBLY
# if SIZEOF_VOID_P==8
#  define cpuid(a, b, c, d, n) asm("xchgq %%rbx, %1; cpuid; xchgq %%rbx, %1": "=a" (a), "=r" (b), "=c" (c), "=d" (d) : "a" (n));
# else
#  define cpuid(a, b, c, d, n) asm("xchgl %%ebx, %1; cpuid; xchgl %%ebx, %1": "=a" (a), "=r" (b), "=c" (c), "=d" (d) : "a" (n));
# endif
#endif

namespace Ingen {
namespace Server {

/** Set flags to disable denormal processing.
 */
inline void
set_denormal_flags()
{
#ifdef USE_ASSEMBLY
#ifdef __SSE__
	unsigned long a, b, c, d0, d1;
	cpuid(a, b, c, d1, 1);
	if (d1 & 1<<25) { /* It has SSE support */
		_MM_SET_FLUSH_ZERO_MODE(_MM_FLUSH_ZERO_ON);
		const int family    = (a >> 8)  & 0xF;
		const int extfamily = (a >> 20) & 0xFF;
		const int model     = (a >> 4)  & 0xF;
		const int stepping  = a         & 0xF;
		cpuid(a, b, c, d0, 0);
		if (b == 0x756e6547) { /* It's an Intel */
			if (family == 15 && extfamily == 0 && model == 0 && stepping < 7) {
				return;
			}
		}
		if (d1 & 1<<26) { /* bit 26, SSE2 support */
			_mm_setcsr(_mm_getcsr() | 0x8040); // set DAZ and FZ bits of MXCSR
			Raul::info << "Set SSE denormal fix flag" << endl;
		}
	} else {
		Raul::warn << "This code has been built with SSE support, but your processor does"
		           << " not support the SSE instruction set, exiting." << std::endl;
		exit(EXIT_FAILURE);
	}
#endif
#endif
}

static inline std::string
ingen_jack_port_name(const Raul::Path& path)
{
	return path.chop_start("/");
}

} // namespace Server
} // namespace Ingen

#endif // INGEN_ENGINE_UTIL_HPP