2013-07-29 06:08:34 +08:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
// Utility functions that depend on Win32. Notably, our memory allocation;
|
|
|
|
// we use two separate allocators, one for long-lived stuff and one for
|
|
|
|
// stuff that gets freed after every regeneration of the model, to save us
|
|
|
|
// the trouble of freeing the latter explicitly.
|
|
|
|
//
|
|
|
|
// Copyright 2008-2013 Jonathan Westhues.
|
|
|
|
//-----------------------------------------------------------------------------
|
2009-04-20 15:30:09 +08:00
|
|
|
#include "solvespace.h"
|
|
|
|
|
2015-03-24 01:49:04 +08:00
|
|
|
namespace SolveSpace {
|
2009-04-20 15:30:09 +08:00
|
|
|
static HANDLE PermHeap, TempHeap;
|
|
|
|
|
2013-08-27 02:58:35 +08:00
|
|
|
void dbp(const char *str, ...)
|
2009-04-20 15:30:09 +08:00
|
|
|
{
|
|
|
|
va_list f;
|
|
|
|
static char buf[1024*50];
|
|
|
|
va_start(f, str);
|
|
|
|
_vsnprintf(buf, sizeof(buf), str, f);
|
|
|
|
va_end(f);
|
|
|
|
|
2015-12-27 15:51:28 +08:00
|
|
|
// The native version of OutputDebugString, unlike most others,
|
|
|
|
// is OutputDebugStringA.
|
|
|
|
OutputDebugStringA(buf);
|
|
|
|
}
|
|
|
|
|
2016-05-19 03:38:17 +08:00
|
|
|
void assert_failure(const char *file, unsigned line, const char *function,
|
|
|
|
const char *condition, const char *message) {
|
|
|
|
dbp("File %s, line %u, function %s:\n", file, line, function);
|
|
|
|
dbp("Assertion '%s' failed: ((%s) == false).\n", message, condition);
|
|
|
|
#ifdef NDEBUG
|
|
|
|
_exit(1);
|
|
|
|
#else
|
|
|
|
abort();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2015-12-27 15:51:28 +08:00
|
|
|
std::string Narrow(const wchar_t *in)
|
|
|
|
{
|
|
|
|
std::string out;
|
|
|
|
DWORD len = WideCharToMultiByte(CP_UTF8, 0, in, -1, NULL, 0, NULL, NULL);
|
|
|
|
out.resize(len - 1);
|
|
|
|
if(!WideCharToMultiByte(CP_UTF8, 0, in, -1, &out[0], len, NULL, NULL))
|
|
|
|
oops();
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string Narrow(const std::wstring &in)
|
|
|
|
{
|
|
|
|
if(in == L"") return "";
|
|
|
|
|
|
|
|
std::string out;
|
|
|
|
out.resize(WideCharToMultiByte(CP_UTF8, 0, &in[0], in.length(), NULL, 0, NULL, NULL));
|
|
|
|
if(!WideCharToMultiByte(CP_UTF8, 0, &in[0], in.length(),
|
|
|
|
&out[0], out.length(), NULL, NULL))
|
|
|
|
oops();
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::wstring Widen(const char *in)
|
|
|
|
{
|
|
|
|
std::wstring out;
|
|
|
|
DWORD len = MultiByteToWideChar(CP_UTF8, 0, in, -1, NULL, 0);
|
|
|
|
out.resize(len - 1);
|
|
|
|
if(!MultiByteToWideChar(CP_UTF8, 0, in, -1, &out[0], len))
|
|
|
|
oops();
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::wstring Widen(const std::string &in)
|
|
|
|
{
|
|
|
|
if(in == "") return L"";
|
|
|
|
|
|
|
|
std::wstring out;
|
|
|
|
out.resize(MultiByteToWideChar(CP_UTF8, 0, &in[0], in.length(), NULL, 0));
|
|
|
|
if(!MultiByteToWideChar(CP_UTF8, 0, &in[0], in.length(), &out[0], out.length()))
|
|
|
|
oops();
|
|
|
|
return out;
|
2009-04-20 15:30:09 +08:00
|
|
|
}
|
|
|
|
|
2015-12-27 16:09:00 +08:00
|
|
|
FILE *ssfopen(const std::string &filename, const char *mode)
|
|
|
|
{
|
|
|
|
// Prepend \\?\ UNC prefix unless already an UNC path.
|
|
|
|
// We never try to fopen paths that are not absolute or
|
|
|
|
// contain separators inappropriate for the platform;
|
|
|
|
// thus, it is always safe to prepend this prefix.
|
|
|
|
std::string uncFilename = filename;
|
|
|
|
if(uncFilename.substr(0, 2) != "\\\\")
|
|
|
|
uncFilename = "\\\\?\\" + uncFilename;
|
|
|
|
|
|
|
|
if(filename.length() != strlen(filename.c_str())) oops();
|
|
|
|
return _wfopen(Widen(uncFilename).c_str(), Widen(mode).c_str());
|
|
|
|
}
|
|
|
|
|
|
|
|
void ssremove(const std::string &filename)
|
|
|
|
{
|
|
|
|
if(filename.length() != strlen(filename.c_str())) oops();
|
|
|
|
_wremove(Widen(filename).c_str());
|
|
|
|
}
|
|
|
|
|
2009-04-20 15:30:09 +08:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
// A separate heap, on which we allocate expressions. Maybe a bit faster,
|
|
|
|
// since no fragmentation issues whatsoever, and it also makes it possible
|
|
|
|
// to be sloppy with our memory management, and just free everything at once
|
|
|
|
// at the end.
|
|
|
|
//-----------------------------------------------------------------------------
|
2013-08-27 04:40:25 +08:00
|
|
|
void *AllocTemporary(size_t n)
|
2009-04-20 15:30:09 +08:00
|
|
|
{
|
|
|
|
void *v = HeapAlloc(TempHeap, HEAP_NO_SERIALIZE | HEAP_ZERO_MEMORY, n);
|
|
|
|
if(!v) oops();
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
void FreeTemporary(void *p) {
|
|
|
|
HeapFree(TempHeap, HEAP_NO_SERIALIZE, p);
|
|
|
|
}
|
|
|
|
void FreeAllTemporary(void)
|
|
|
|
{
|
|
|
|
if(TempHeap) HeapDestroy(TempHeap);
|
|
|
|
TempHeap = HeapCreate(HEAP_NO_SERIALIZE, 1024*1024*20, 0);
|
|
|
|
// This is a good place to validate, because it gets called fairly
|
|
|
|
// often.
|
|
|
|
vl();
|
|
|
|
}
|
|
|
|
|
2013-08-27 04:40:25 +08:00
|
|
|
void *MemAlloc(size_t n) {
|
2009-04-20 15:30:09 +08:00
|
|
|
void *p = HeapAlloc(PermHeap, HEAP_NO_SERIALIZE | HEAP_ZERO_MEMORY, n);
|
|
|
|
if(!p) oops();
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
void MemFree(void *p) {
|
|
|
|
HeapFree(PermHeap, HEAP_NO_SERIALIZE, p);
|
|
|
|
}
|
|
|
|
|
|
|
|
void vl(void) {
|
|
|
|
if(!HeapValidate(TempHeap, HEAP_NO_SERIALIZE, NULL)) oops();
|
|
|
|
if(!HeapValidate(PermHeap, HEAP_NO_SERIALIZE, NULL)) oops();
|
|
|
|
}
|
|
|
|
|
|
|
|
void InitHeaps(void) {
|
|
|
|
// Create the heap used for long-lived stuff (that gets freed piecewise).
|
|
|
|
PermHeap = HeapCreate(HEAP_NO_SERIALIZE, 1024*1024*20, 0);
|
|
|
|
// Create the heap that we use to store Exprs and other temp stuff.
|
|
|
|
FreeAllTemporary();
|
|
|
|
}
|
2015-11-06 19:11:11 +08:00
|
|
|
}
|