2016-11-24 13:05:06 +00:00
|
|
|
#ifndef slic3r_Utils_hpp_
|
|
|
|
#define slic3r_Utils_hpp_
|
|
|
|
|
2017-10-26 15:17:39 +00:00
|
|
|
#include <locale>
|
2018-11-02 09:31:10 +00:00
|
|
|
#include <utility>
|
|
|
|
#include <functional>
|
2018-11-08 14:19:50 +00:00
|
|
|
#include <type_traits>
|
2017-10-26 15:17:39 +00:00
|
|
|
|
2018-03-13 11:39:57 +00:00
|
|
|
#include "libslic3r.h"
|
|
|
|
|
2016-11-24 13:05:06 +00:00
|
|
|
namespace Slic3r {
|
|
|
|
|
|
|
|
extern void set_logging_level(unsigned int level);
|
2017-03-03 11:53:05 +00:00
|
|
|
extern void trace(unsigned int level, const char *message);
|
2018-12-18 13:10:31 +00:00
|
|
|
// Format memory allocated, separate thousands by comma.
|
|
|
|
extern std::string format_memsize_MB(size_t n);
|
2018-12-18 10:31:41 +00:00
|
|
|
// Return string to be added to the boost::log output to inform about the current process memory allocation.
|
|
|
|
// The string is non-empty only if the loglevel >= info (3).
|
|
|
|
extern std::string log_memory_info();
|
2018-09-17 13:12:13 +00:00
|
|
|
extern void disable_multi_threading();
|
2016-11-24 13:05:06 +00:00
|
|
|
|
2017-10-17 18:00:15 +00:00
|
|
|
// Set a path with GUI resource files.
|
|
|
|
void set_var_dir(const std::string &path);
|
2017-10-25 10:53:31 +00:00
|
|
|
// Return a full path to the GUI resource files.
|
2017-10-17 18:00:15 +00:00
|
|
|
const std::string& var_dir();
|
2017-10-25 10:53:31 +00:00
|
|
|
// Return a full resource path for a file_name.
|
2017-10-17 18:00:15 +00:00
|
|
|
std::string var(const std::string &file_name);
|
|
|
|
|
2017-12-10 12:19:44 +00:00
|
|
|
// Set a path with various static definition data (for example the initial config bundles).
|
|
|
|
void set_resources_dir(const std::string &path);
|
|
|
|
// Return a full path to the resources directory.
|
|
|
|
const std::string& resources_dir();
|
2018-02-12 07:57:32 +00:00
|
|
|
|
|
|
|
// Set a path with GUI localization files.
|
|
|
|
void set_local_dir(const std::string &path);
|
2018-02-07 16:13:52 +00:00
|
|
|
// Return a full path to the localization directory.
|
2018-02-12 07:57:32 +00:00
|
|
|
const std::string& localization_dir();
|
2017-12-10 12:19:44 +00:00
|
|
|
|
2017-10-25 10:53:31 +00:00
|
|
|
// Set a path with preset files.
|
|
|
|
void set_data_dir(const std::string &path);
|
|
|
|
// Return a full path to the GUI resource files.
|
|
|
|
const std::string& data_dir();
|
|
|
|
|
2017-12-21 15:56:33 +00:00
|
|
|
// A special type for strings encoded in the local Windows 8-bit code page.
|
|
|
|
// This type is only needed for Perl bindings to relay to Perl that the string is raw, not UTF-8 encoded.
|
|
|
|
typedef std::string local_encoded_string;
|
|
|
|
|
|
|
|
// Convert an UTF-8 encoded string into local coding.
|
|
|
|
// On Windows, the UTF-8 string is converted to a local 8-bit code page.
|
|
|
|
// On OSX and Linux, this function does no conversion and returns a copy of the source string.
|
|
|
|
extern local_encoded_string encode_path(const char *src);
|
2017-08-03 15:31:31 +00:00
|
|
|
extern std::string decode_path(const char *src);
|
|
|
|
extern std::string normalize_utf8_nfc(const char *src);
|
|
|
|
|
2018-09-14 07:28:00 +00:00
|
|
|
// Safely rename a file even if the target exists.
|
|
|
|
// On Windows, the file explorer (or anti-virus or whatever else) often locks the file
|
|
|
|
// for a short while, so the file may not be movable. Retry while we see recoverable errors.
|
|
|
|
extern int rename_file(const std::string &from, const std::string &to);
|
|
|
|
|
|
|
|
// Copy a file, adjust the access attributes, so that the target is writable.
|
|
|
|
extern int copy_file(const std::string &from, const std::string &to);
|
|
|
|
|
2017-12-21 15:56:33 +00:00
|
|
|
// File path / name / extension splitting utilities, working with UTF-8,
|
|
|
|
// to be published to Perl.
|
|
|
|
namespace PerlUtils {
|
|
|
|
// Get a file name including the extension.
|
|
|
|
extern std::string path_to_filename(const char *src);
|
|
|
|
// Get a file name without the extension.
|
|
|
|
extern std::string path_to_stem(const char *src);
|
|
|
|
// Get just the extension.
|
|
|
|
extern std::string path_to_extension(const char *src);
|
|
|
|
// Get a directory without the trailing slash.
|
|
|
|
extern std::string path_to_parent_path(const char *src);
|
|
|
|
};
|
|
|
|
|
2018-11-19 10:07:39 +00:00
|
|
|
std::string string_printf(const char *format, ...);
|
|
|
|
|
2017-10-30 17:15:41 +00:00
|
|
|
// Timestamp formatted for header_slic3r_generated().
|
|
|
|
extern std::string timestamp_str();
|
|
|
|
// Standard "generated by Slic3r version xxx timestamp xxx" header string,
|
|
|
|
// to be placed at the top of Slic3r generated files.
|
2017-11-01 18:30:05 +00:00
|
|
|
inline std::string header_slic3r_generated() { return std::string("generated by " SLIC3R_FORK_NAME " " SLIC3R_VERSION " " ) + timestamp_str(); }
|
2017-10-30 17:15:41 +00:00
|
|
|
|
2018-04-20 09:05:00 +00:00
|
|
|
// getpid platform wrapper
|
|
|
|
extern unsigned get_current_pid();
|
|
|
|
|
2018-09-21 08:18:56 +00:00
|
|
|
template <typename Real>
|
|
|
|
Real round_nearest(Real value, unsigned int decimals)
|
|
|
|
{
|
|
|
|
Real res = (Real)0;
|
|
|
|
if (decimals == 0)
|
|
|
|
res = ::round(value);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Real power = ::pow((Real)10, (int)decimals);
|
|
|
|
res = ::round(value * power + (Real)0.5) / power;
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2017-03-28 15:09:57 +00:00
|
|
|
// Compute the next highest power of 2 of 32-bit v
|
|
|
|
// http://graphics.stanford.edu/~seander/bithacks.html
|
2018-09-25 09:53:05 +00:00
|
|
|
inline uint16_t next_highest_power_of_2(uint16_t v)
|
2017-03-28 15:09:57 +00:00
|
|
|
{
|
2018-09-25 09:53:05 +00:00
|
|
|
if (v != 0)
|
|
|
|
-- v;
|
2017-03-28 15:09:57 +00:00
|
|
|
v |= v >> 1;
|
|
|
|
v |= v >> 2;
|
|
|
|
v |= v >> 4;
|
2018-09-25 09:53:05 +00:00
|
|
|
v |= v >> 8;
|
|
|
|
return ++ v;
|
|
|
|
}
|
|
|
|
inline uint32_t next_highest_power_of_2(uint32_t v)
|
|
|
|
{
|
|
|
|
if (v != 0)
|
|
|
|
-- v;
|
|
|
|
v |= v >> 1;
|
|
|
|
v |= v >> 2;
|
|
|
|
v |= v >> 4;
|
|
|
|
v |= v >> 8;
|
|
|
|
v |= v >> 16;
|
|
|
|
return ++ v;
|
|
|
|
}
|
|
|
|
inline uint64_t next_highest_power_of_2(uint64_t v)
|
|
|
|
{
|
|
|
|
if (v != 0)
|
|
|
|
-- v;
|
|
|
|
v |= v >> 1;
|
|
|
|
v |= v >> 2;
|
|
|
|
v |= v >> 4;
|
|
|
|
v |= v >> 8;
|
|
|
|
v |= v >> 16;
|
|
|
|
v |= v >> 32;
|
2017-03-28 15:09:57 +00:00
|
|
|
return ++ v;
|
|
|
|
}
|
|
|
|
|
2018-11-08 14:19:50 +00:00
|
|
|
// On some implementations (such as some versions of clang), the size_t is a type of its own, so we need to overload for size_t.
|
|
|
|
// Typically, though, the size_t type aliases to uint64_t / uint32_t.
|
|
|
|
// We distinguish that here and provide implementation for size_t if and only if it is a distinct type
|
|
|
|
template<class T> size_t next_highest_power_of_2(T v,
|
|
|
|
typename std::enable_if<std::is_same<T, size_t>::value, T>::type = 0, // T is size_t
|
|
|
|
typename std::enable_if<!std::is_same<T, uint64_t>::value, T>::type = 0, // T is not uint64_t
|
|
|
|
typename std::enable_if<!std::is_same<T, uint32_t>::value, T>::type = 0, // T is not uint32_t
|
|
|
|
typename std::enable_if<sizeof(T) == 8, T>::type = 0) // T is 64 bits
|
|
|
|
{
|
2018-09-25 10:43:53 +00:00
|
|
|
return next_highest_power_of_2(uint64_t(v));
|
2018-11-08 14:19:50 +00:00
|
|
|
}
|
|
|
|
template<class T> size_t next_highest_power_of_2(T v,
|
|
|
|
typename std::enable_if<std::is_same<T, size_t>::value, T>::type = 0, // T is size_t
|
|
|
|
typename std::enable_if<!std::is_same<T, uint64_t>::value, T>::type = 0, // T is not uint64_t
|
|
|
|
typename std::enable_if<!std::is_same<T, uint32_t>::value, T>::type = 0, // T is not uint32_t
|
|
|
|
typename std::enable_if<sizeof(T) == 4, T>::type = 0) // T is 32 bits
|
|
|
|
{
|
2018-09-25 11:19:50 +00:00
|
|
|
return next_highest_power_of_2(uint32_t(v));
|
2018-09-25 10:43:53 +00:00
|
|
|
}
|
2018-11-08 14:19:50 +00:00
|
|
|
|
2018-09-25 10:43:53 +00:00
|
|
|
|
2018-07-23 12:39:50 +00:00
|
|
|
extern std::string xml_escape(std::string text);
|
|
|
|
|
2018-11-02 09:31:10 +00:00
|
|
|
|
|
|
|
class ScopeGuard
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
typedef std::function<void()> Closure;
|
|
|
|
private:
|
|
|
|
bool committed;
|
|
|
|
Closure closure;
|
|
|
|
|
|
|
|
public:
|
|
|
|
ScopeGuard() {}
|
|
|
|
ScopeGuard(Closure closure) : closure(std::move(closure)) {}
|
|
|
|
ScopeGuard(const ScopeGuard&) = delete;
|
|
|
|
ScopeGuard(ScopeGuard &&other) : closure(std::move(other.closure)) {}
|
|
|
|
|
|
|
|
~ScopeGuard()
|
|
|
|
{
|
|
|
|
if (closure) { closure(); }
|
|
|
|
}
|
|
|
|
|
|
|
|
ScopeGuard& operator=(const ScopeGuard&) = delete;
|
|
|
|
ScopeGuard& operator=(ScopeGuard &&other)
|
|
|
|
{
|
|
|
|
closure = std::move(other.closure);
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
void reset() { closure = Closure(); }
|
|
|
|
};
|
|
|
|
|
2016-11-24 13:05:06 +00:00
|
|
|
} // namespace Slic3r
|
|
|
|
|
2018-12-18 14:55:45 +00:00
|
|
|
#if WIN32
|
|
|
|
#define SLIC3R_STDVEC_MEMSIZE(NAME, TYPE) NAME.capacity() * ((sizeof(TYPE) + __alignof(TYPE) - 1) / __alignof(TYPE)) * __alignof(TYPE)
|
|
|
|
#else
|
|
|
|
#define SLIC3R_STDVEC_MEMSIZE(NAME, TYPE) NAME.capacity() * ((sizeof(TYPE) + alignof(TYPE) - 1) / alignof(TYPE)) * alignof(TYPE)
|
|
|
|
#endif
|
|
|
|
|
2016-11-24 13:05:06 +00:00
|
|
|
#endif // slic3r_Utils_hpp_
|