16 #define GETTEXT_DOMAIN "wesnoth-lib"
29 #include "addon/manager.hpp"
38 #include <SDL2/SDL_image.h>
39 #include <SDL2/SDL_mixer.h>
41 #include <boost/algorithm/string.hpp>
42 #include <boost/predef.h>
43 #include <boost/version.hpp>
46 #include <openssl/crypto.h>
47 #include <openssl/opensslv.h>
50 #include <curl/curl.h>
52 #include <pango/pangocairo.h>
57 #include <CoreFoundation/CoreFoundation.h>
65 struct version_table_manager
70 version_table_manager();
73 const version_table_manager versions;
75 std::string format_version(
unsigned a,
unsigned b,
unsigned c)
80 std::string format_version(
const SDL_version& v)
82 return formatter() <<
static_cast<unsigned>(v.major) <<
'.'
83 <<
static_cast<unsigned>(v.minor) <<
'.'
84 <<
static_cast<unsigned>(v.patch);
89 std::string format_openssl_patch_level(uint8_t
p)
92 ? std::string(1,
'a' +
static_cast<char>(
p) - 1)
93 :
"patch" + std::to_string(
p);
96 std::string format_openssl_version(
long v)
98 int major, minor, fix, patch, status;
99 std::ostringstream fmt;
109 minor = v & 0x0F00L >> 8;
110 fix = v & 0x00F0L >> 4;
113 fmt <<
"0." << minor <<
'.' << fix;
115 fmt << format_openssl_patch_level(patch);
124 major = (v & 0xF0000000L) >> 28;
125 minor = (v & 0x0FF00000L) >> 20;
126 fix = (v & 0x000FF000L) >> 12;
127 patch = (v & 0x00000FF0L) >> 4;
128 status = (v & 0x0000000FL);
130 if(v < 0x00905100L) {
142 const uint8_t is_final = (v & 0xF00L) >> 8;
143 status = is_final ? 0xF : 0;
145 }
else if(v < 0x00906000L) {
158 fmt << major <<
'.' << minor <<
'.' << fix;
161 fmt << format_openssl_patch_level(patch);
166 }
else if(status < 0xF) {
167 fmt <<
"-beta" << status;
176 version_table_manager::version_table_manager()
182 SDL_version sdl_version;
188 SDL_VERSION(&sdl_version);
191 SDL_GetVersion(&sdl_version);
200 SDL_IMAGE_VERSION(&sdl_version);
203 const SDL_version* sdl_rt_version = IMG_Linked_Version();
214 SDL_MIXER_VERSION(&sdl_version);
217 sdl_rt_version = Mix_Linked_Version();
236 compiled[
LIB_LUA] = LUA_VERSION_MAJOR
"." LUA_VERSION_MINOR
"." LUA_VERSION_RELEASE;
254 (LIBCURL_VERSION_NUM & 0xFF0000) >> 16,
255 (LIBCURL_VERSION_NUM & 0x00FF00) >> 8,
256 LIBCURL_VERSION_NUM & 0x0000FF);
257 curl_version_info_data *curl_ver = curl_version_info(CURLVERSION_NOW);
258 if(curl_ver && curl_ver->version) {
285 features.emplace_back(
N_(
"feature^Lua console completion"));
292 features.emplace_back(
N_(
"feature^D-Bus notifications back end"));
301 features.emplace_back(
N_(
"feature^Win32 notifications back end"));
307 features.emplace_back(
N_(
"feature^Cocoa notifications back end"));
312 const std::string empty_version =
"";
318 #if BOOST_ARCH_X86_64
320 #elif BOOST_ARCH_X86_32
322 #elif BOOST_ARCH_ARM && (defined(__arm64) || defined(_M_ARM64))
326 #elif BOOST_ARCH_IA64
330 #elif BOOST_ARCH_ALPHA
332 #elif BOOST_ARCH_MIPS
334 #elif BOOST_ARCH_SPARC
337 #warning Unrecognized platform or Boost.Predef broken/unavailable
346 return _(
"cpu_architecture^<unknown>");
352 std::vector<optional_feature> res = versions.features;
354 for(std::size_t k = 0; k < res.size(); ++k) {
356 res[k].name =
_(res[k].name.c_str());
360 const auto caret_pos = res[k].name.find(
'^');
361 if(caret_pos != std::string::npos) {
362 res[k].name.erase(0, caret_pos + 1);
372 return empty_version;
375 return versions.compiled[lib];
381 return empty_version;
384 return versions.linked[lib];
390 return empty_version;
393 return versions.names[lib];
400 if(infofile.is_open()) {
401 std::getline(infofile,
info);
421 using list_entry = std::pair<std::string, std::string>;
422 using contents_list = std::vector<list_entry>;
424 list_formatter(
const std::string& heading,
const contents_list& contents = {},
const std::string& empty_placeholder =
"")
431 void insert(
const std::string&
label,
const std::string& value)
436 void set_placeholder(
const std::string& placeholder)
441 void stream_put(std::ostream& os)
const;
456 void list_formatter::stream_put(std::ostream& os)
const
465 auto label_length_comparator = [](
const list_entry& a,
const list_entry&
b)
470 const auto longest_entry_label = std::max_element(
contents_.begin(),
contents_.end(), label_length_comparator);
471 const std::size_t min_length = longest_entry_label !=
contents_.end()
476 const std::size_t prev_width = os.width();
477 const std::ostream::fmtflags prev_flags = os.flags();
482 os << std::setw(min_length) << entry.first +
label_delimiter << entry.second <<
'\n';
485 os.width(prev_width);
486 os.flags(prev_flags);
492 std::ostream&
operator<<(std::ostream& os,
const list_formatter& fmt)
498 list_formatter library_versions_report_internal(
const std::string& heading =
"")
500 list_formatter fmt{heading};
504 if(versions.names[
n].empty()) {
508 std::string text = versions.compiled[
n];
509 if(!versions.linked[
n].empty()) {
510 text +=
" (runtime " + versions.linked[
n] +
")";
513 fmt.insert(versions.names[
n], text);
519 list_formatter optional_features_report_internal(
const std::string& heading =
"")
521 list_formatter fmt{heading};
525 for(
const auto& feature :
features) {
526 fmt.insert(feature.name, feature.enabled ?
"yes" :
"no");
532 inline std::string geometry_to_string(
point p)
534 return std::to_string(
p.x) +
'x' + std::to_string(
p.y);
537 template<
typename coordinateType>
538 inline std::string geometry_to_string(coordinateType horizontal, coordinateType vertical)
541 return formatter() << std::fixed << std::setprecision(2) << horizontal <<
'x' << vertical;
544 std::string format_sdl_driver_list(std::vector<std::string> drivers,
const std::string&
current_driver)
546 bool found_current_driver =
false;
548 for(
auto& drvname : drivers) {
550 found_current_driver =
true;
555 if(drivers.empty() || !found_current_driver) {
563 list_formatter video_settings_report_internal(
const std::string& heading =
"")
565 list_formatter fmt{heading};
567 std::string placeholder;
570 placeholder =
"Running in non-interactive mode.";
574 placeholder =
"Video not initialized yet.";
577 if(!placeholder.empty()) {
578 fmt.set_placeholder(placeholder);
586 std::string dpi_report;
588 dpi_report = dpi.first == 0.0f || dpi.second == 0.0f ?
590 geometry_to_string(dpi.first, dpi.second);
592 fmt.insert(
"SDL video drivers", format_sdl_driver_list(drivers,
current_driver));
597 fmt.insert(
"Screen dpi", dpi_report);
602 fmt.insert(
info.first,
info.second);
608 list_formatter sound_settings_report_internal(
const std::string& heading =
"")
610 list_formatter fmt{heading};
614 if(!driver_status.initialized) {
615 fmt.set_placeholder(
"Audio not initialized.");
622 static std::map<uint16_t, std::string> audio_format_names = {
624 { AUDIO_U8,
"unsigned 8 bit" },
625 { AUDIO_S8,
"signed 8 bit" },
627 { AUDIO_U16LSB,
"unsigned 16 bit little-endian" },
628 { AUDIO_U16MSB,
"unsigned 16 bit big-endian" },
629 { AUDIO_S16LSB,
"signed 16 bit little-endian" },
630 { AUDIO_S16MSB,
"signed 16 bit big-endian" },
632 { AUDIO_S32LSB,
"signed 32 bit little-endian" },
633 { AUDIO_S32MSB,
"signed 32 bit big-endian" },
634 { AUDIO_F32LSB,
"signed 32 bit floating point little-endian" },
635 { AUDIO_F32MSB,
"signed 32 bit floating point big-endian" },
638 auto fmt_names_it = audio_format_names.find(driver_status.format);
640 const std::string fmt_name = fmt_names_it != audio_format_names.end()
641 ? fmt_names_it->second
642 :
formatter() <<
"0x" << std::setfill(
'0') << std::setw(2*
sizeof(driver_status.format)) << std::hex << std::uppercase << driver_status.format;
644 fmt.insert(
"SDL audio drivers", format_sdl_driver_list(drivers,
current_driver));
645 fmt.insert(
"Number of channels", std::to_string(driver_status.channels));
646 fmt.insert(
"Output rate", std::to_string(driver_status.frequency) +
" Hz");
647 fmt.insert(
"Sample format", fmt_name);
648 fmt.insert(
"Sample size", std::to_string(driver_status.chunk_size) +
" bytes");
657 return formatter{} << library_versions_report_internal();
662 return formatter{} << optional_features_report_internal();
667 list_formatter::contents_list paths{
677 for(
auto& entry : paths) {
681 list_formatter::contents_list addons;
687 std::ostringstream o;
693 << list_formatter{
"Game paths", paths}
694 << library_versions_report_internal(
"Libraries")
695 << optional_features_report_internal(
"Features")
696 << video_settings_report_internal(
"Current video settings")
697 << sound_settings_report_internal(
"Current audio settings")
698 << list_formatter(
"Installed add-ons", addons,
"No add-ons installed.");
std::ostream & operator<<(std::ostream &s, const ai::attack_result &r)
std::map< std::string, std::string > installed_addons_and_versions()
Retrieves the ids and versions of all installed add-ons.
static const char heading_delimiter
std::vector< optional_feature > features
std::vector< std::string > compiled
std::vector< std::string > linked
std::vector< std::string > names
static const std::string label_delimiter
Platform identification and version information functions.
Declarations for File-IO.
Interfaces for manipulating version numbers of engine, add-ons, etc.
static std::string _(const char *str)
std::string label
What to show in the filter's drop-down list.
T end(const std::pair< T, T > &p)
T begin(const std::pair< T, T > &p)
std::string os_version()
Returns a string with the running OS name and version information.
std::string get_cache_dir()
std::string get_user_data_dir()
std::string get_saves_dir()
std::string get_logs_dir()
std::string get_addons_dir()
std::string sanitize_path(const std::string &path)
Sanitizes a path to remove references to the user's name.
Game configuration data as global variables.
const std::string & library_name(LIBRARY_ID lib)
Retrieve the user-visible name for the given library.
std::string full_build_report()
Produce a bug report-style info dump.
std::string library_versions_report()
Produce a plain-text report of library versions suitable for stdout/stderr.
std::vector< optional_feature > optional_features_table(bool localize)
Retrieve the features table.
const std::string & library_runtime_version(LIBRARY_ID lib)
Retrieve the runtime version number of the given library.
const std::string & library_build_version(LIBRARY_ID lib)
Retrieve the build-time version number of the given library.
std::string build_arch()
Obtain the processor architecture for this build.
std::string optional_features_report()
Produce a plain-text report of features suitable for stdout/stderr.
const std::string revision
std::string dist_channel_id()
Return the distribution channel identifier, or "Default" if missing.
std::vector< std::string > enumerate_drivers()
std::string current_driver()
std::string & insert(std::string &str, const std::size_t pos, const std::string &insert)
Insert a UTF-8 string at the specified position.
std::size_t size(const std::string &str)
Length in characters of a UTF-8 string.
void trim(std::string_view &s)
std::string join(const T &v, const std::string &s=",")
Generates a new string joining container items in a list.
bool headless()
The game is running headless.
std::vector< std::pair< std::string, std::string > > renderer_report()
Provides diagnostic information about the current renderer for the build_info API.
point output_size()
Returns the size of the final render target.
point game_canvas_size()
The size of the game canvas, in drawing coordinates / game pixels.
bool has_window()
Whether the game has set up a window to render into.
std::vector< std::string > enumerate_drivers()
A list of available video drivers.
std::pair< float, float > get_dpi()
Retrieves the current game screen DPI for the build_info API.
point current_resolution()
The current window size in desktop coordinates.
int current_refresh_rate()
The refresh rate of the screen.
std::string current_driver()
The current video driver in use, or else "<not initialized>".
static driver_status query()
static map_location::direction n