You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

1378 lines
37 KiB

/* /////////////////////////////////////////////////////////////////////////
* File: winstl/system/version_info.hpp (originally in MWVerInf.h, ::SynesisWin)
*
* Purpose: Helper for accessing version information.
*
* Created: 16th February 1998
* Updated: 19th May 2010
*
* Home: http://stlsoft.org/
*
* Copyright (c) 1998-2010, Matthew Wilson and Synesis Software
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* - Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
* - Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
* - Neither the name(s) of Matthew Wilson and Synesis Software nor the names of
* any contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*
* ////////////////////////////////////////////////////////////////////// */
/** \file winstl/system/version_info.hpp
*
* \brief [C++ only] Definition of the winstl::version_info class
* template
* (\ref group__library__system "System" Library).
*/
#ifndef WINSTL_INCL_WINSTL_SYSTEM_HPP_VERSION_INFO
#define WINSTL_INCL_WINSTL_SYSTEM_HPP_VERSION_INFO
#ifndef STLSOFT_DOCUMENTATION_SKIP_SECTION
# define WINSTL_VER_WINSTL_SYSTEM_HPP_VERSION_INFO_MAJOR 5
# define WINSTL_VER_WINSTL_SYSTEM_HPP_VERSION_INFO_MINOR 2
# define WINSTL_VER_WINSTL_SYSTEM_HPP_VERSION_INFO_REVISION 8
# define WINSTL_VER_WINSTL_SYSTEM_HPP_VERSION_INFO_EDIT 126
#endif /* !STLSOFT_DOCUMENTATION_SKIP_SECTION */
/* /////////////////////////////////////////////////////////////////////////
* Includes
*/
#ifndef WINSTL_INCL_WINSTL_H_WINSTL
# include <winstl/winstl.h>
#endif /* !WINSTL_INCL_WINSTL_H_WINSTL */
#ifndef WINSTL_INCL_WINSTL_ERROR_HPP_WINDOWS_EXCEPTIONS
# include <winstl/error/exceptions.hpp>
#endif /* !WINSTL_INCL_WINSTL_ERROR_HPP_WINDOWS_EXCEPTIONS */
#if defined(STLSOFT_COMPILER_IS_MSVC) && \
_MSC_VER < 1200
# if defined(UNICODE) || \
defined(_UNICODE)
# error winstl::version_info is not supported on Visual C++ 5.0 (or previous) with UNICODE compilations
# endif /* Unicode */
# define WINSTL_VERSION_INFO_NO_USE_FILE_PATH_BUFFER
#else /* ? compiler */
# ifndef WINSTL_INCL_WINSTL_FILESYSTEM_HPP_FILE_PATH_BUFFER
# include <winstl/filesystem/file_path_buffer.hpp>
# endif /* !WINSTL_INCL_WINSTL_FILESYSTEM_HPP_FILE_PATH_BUFFER */
#endif /* compiler */
#ifndef WINSTL_INCL_WINSTL_MEMORY_HPP_PROCESSHEAP_ALLOCATOR
# include <winstl/memory/processheap_allocator.hpp>
#endif /* !WINSTL_INCL_WINSTL_MEMORY_HPP_PROCESSHEAP_ALLOCATOR */
#ifndef STLSOFT_INCL_STLSOFT_CONVERSION_HPP_SAP_CAST
# include <stlsoft/conversion/sap_cast.hpp>
#endif /* !STLSOFT_INCL_STLSOFT_CONVERSION_HPP_SAP_CAST */
#ifndef STLSOFT_INCL_STLSOFT_SHIMS_ACCESS_HPP_STRING
# include <stlsoft/shims/access/string.hpp>
#endif /* !STLSOFT_INCL_STLSOFT_SHIMS_ACCESS_HPP_STRING */
#ifndef STLSOFT_INCL_STLSOFT_UTIL_STD_HPP_ITERATOR_HELPER
# include <stlsoft/util/std/iterator_helper.hpp>
#endif /* !STLSOFT_INCL_STLSOFT_UTIL_STD_HPP_ITERATOR_HELPER */
#ifndef STLSOFT_INCL_STLSOFT_COLLECTIONS_UTIL_HPP_COLLECTIONS
# include <stlsoft/collections/util/collections.hpp>
#endif /* !STLSOFT_INCL_STLSOFT_COLLECTIONS_UTIL_HPP_COLLECTIONS */
#ifdef STLSOFT_CF_EXCEPTION_SUPPORT
# include <stdexcept> // for std::exception
#endif /* STLSOFT_CF_EXCEPTION_SUPPORT */
#ifndef STLSOFT_INCL_H_WCHAR
# define STLSOFT_INCL_H_WCHAR
# include <wchar.h>
#endif /* !STLSOFT_INCL_H_WCHAR */
#ifdef STLSOFT_UNITTEST
# include <stdio.h>
#endif /* STLSOFT_UNITTEST */
/* /////////////////////////////////////////////////////////////////////////
* Namespace
*/
#ifndef _WINSTL_NO_NAMESPACE
# if defined(_STLSOFT_NO_NAMESPACE) || \
defined(STLSOFT_DOCUMENTATION_SKIP_SECTION)
/* There is no stlsoft namespace, so must define ::winstl */
namespace winstl
{
# else
/* Define stlsoft::winstl_project */
namespace stlsoft
{
namespace winstl_project
{
# endif /* _STLSOFT_NO_NAMESPACE */
#endif /* !_WINSTL_NO_NAMESPACE */
/* /////////////////////////////////////////////////////////////////////////
* Structure headers
*/
#ifndef STLSOFT_DOCUMENTATION_SKIP_SECTION
// The following bizarre construction is solely to avoid an ICE in VC6
// that only raises its ugly head in large, non-trivial, compilation
// contexts.
template<ss_typename_param_k T>
struct hdr
{
template<int N>
struct hdr_
{
WORD wLength;
WORD wValueLength;
WORD wType;
WCHAR szKey[1];
};
}; // namespace hdr
typedef hdr<int>::hdr_<1> VS_VERSIONINFO_hdr;
typedef hdr<int>::hdr_<2> StringFileInfo_hdr;
typedef hdr<int>::hdr_<3> VarFileInfo_hdr;
typedef hdr<int>::hdr_<4> Var_hdr;
typedef hdr<int>::hdr_<5> StringTable_hdr;
typedef hdr<int>::hdr_<6> String_hdr;
template<ss_typename_param_k T>
T* rounded_ptr(T* p, ss_size_t n)
{
union
{
T* p;
ss_size_t cb;
} u;
u.p = p;
u.cb = ((n - 1) + u.cb) & ~(n- 1);
WINSTL_ASSERT(ptr_byte_diff(u.p, p) >= 0);
return u.p;
}
template<ss_typename_param_k T>
T* rounded_ptr(T* p, ss_ptrdiff_t byteOffset, ss_size_t n)
{
// 1. This has to be done in a ridiculously long-hand fashion because Borland is a *very* stupid compiler!
#if defined(STLSOFT_COMPILER_IS_BORLAND)
void const* pv = &byteOffset[(char*)p];
#else /* ? compiler */
void const* pv = ptr_byte_offset(p, byteOffset);
#endif /* compiler */
WINSTL_ASSERT(((char*)pv - (char*)p) == byteOffset);
T* p_ = static_cast<T*>(pv);
T* r = rounded_ptr(p_, n);
#ifdef STLSOFT_COMPILER_IS_BORLAND
STLSOFT_SUPPRESS_UNUSED(p);
STLSOFT_SUPPRESS_UNUSED(byteOffset);
#endif /* compiler */
WINSTL_ASSERT(ptr_byte_diff(r, p_) >= 0);
return r;
}
#endif /* !STLSOFT_DOCUMENTATION_SKIP_SECTION */
/* /////////////////////////////////////////////////////////////////////////
* Classes
*/
#ifdef STLSOFT_CF_EXCEPTION_SUPPORT
/** \brief Exception thrown by the version_info class.
*
* \ingroup group__library__system
*/
class version_info_exception
: public windows_exception
{
/// \name Member Types
/// @{
public:
typedef windows_exception parent_class_type;
typedef version_info_exception class_type;
/// @}
/// \name Construction
/// @{
public:
version_info_exception(char const* reason, error_code_type err)
: parent_class_type(reason, err)
{}
/// @}
};
#endif /* STLSOFT_CF_EXCEPTION_SUPPORT */
/** \brief Represents the fixed part of a version information block
*
* \ingroup group__library__system
*/
class fixed_file_info
{
typedef fixed_file_info class_type;
public:
/// Constructor
fixed_file_info(VS_FIXEDFILEINFO const* ffi);
public:
ws_uint16_t ApiVerHigh() const;
ws_uint16_t ApiVerLow() const;
/// The major part of the file version
ws_uint16_t FileVerMajor() const;
/// The minor part of the file version
ws_uint16_t FileVerMinor() const;
/// The revision part of the file version
ws_uint16_t FileVerRevision() const;
/// The build increment part of the file version
ws_uint16_t FileVerBuild() const;
/// The major part of the product version
ws_uint16_t ProductVerMajor() const;
/// The minor part of the product version
ws_uint16_t ProductVerMinor() const;
/// The revision part of the product version
ws_uint16_t ProductVerRevision() const;
/// The build increment part of the product version
ws_uint16_t ProductVerBuild() const;
/// The file flags mask
ws_uint32_t FileFlagsMask() const;
/// The file flags
ws_uint32_t FileFlags() const;
/// The file operating system
ws_uint32_t FileOS() const;
/// The file type
ws_uint32_t FileType() const;
/// The file subtype
ws_uint32_t FileSubtype() const;
/// The timestamp of the file
FILETIME const& FileDateTime() const;
private:
static FILETIME calc_FileDateTime_(VS_FIXEDFILEINFO const* ffi);
private:
VS_FIXEDFILEINFO const* const m_ffi;
FILETIME const m_fileDateTime;
private:
class_type& operator =(class_type const&);
};
/** \brief Represents a variable file part of a version information block
*
* \ingroup group__library__system
*/
class VsVar
{
public:
/// This type
typedef VsVar class_type;
/// Represents a language/code-page pair
struct LangCodePage
{
/// The language
ss_uint16_t language;
/// The code-page
ss_uint16_t codePage;
};
public:
/// Constructor
VsVar(Var_hdr const* p);
/// The length of the variable
ss_size_t length() const;
/// Subscript operator
LangCodePage const& operator [](ss_size_t index) const;
private:
Var_hdr const* m_p;
LangCodePage const* m_values;
};
/** \brief Represents a string part of a version information block
*
* \ingroup group__library__system
*/
class VsString
{
public:
/// This type
typedef VsString class_type;
public:
/// Constructor
VsString(String_hdr const* p);
/// The name of the variable
wchar_t const* name() const;
/// The value of the variable
wchar_t const* value() const;
private:
wchar_t const* m_name;
wchar_t const* m_value;
};
/** \brief Represents a string table
*
* \ingroup group__library__system
*/
class VsStringTable
: public stlsoft_ns_qual(stl_collection_tag)
{
public:
/// This type
typedef VsStringTable class_type;
/// The value type
typedef VsString value_type;
public:
/// Constructor
VsStringTable(StringTable_hdr const* p);
/// The key
wchar_t const* Key() const;
/// The non-mutating (const) iterator
class const_iterator
: public stlsoft_ns_qual(iterator_base)<winstl_ns_qual_std(forward_iterator_tag)
, value_type
, ws_ptrdiff_t
, void // By-Value Temporary reference
, value_type // By-Value Temporary reference
>
{
public:
/// This type
typedef const_iterator class_type;
/// The value type
typedef VsString value_type;
public:
/// Constructor
const_iterator(void const* p);
/// Pre-increment operator
class_type& operator ++();
/// Post-increment operator
class_type operator ++(int);
value_type operator *() const;
ws_bool_t operator ==(class_type const& rhs) const;
ws_bool_t operator !=(class_type const& rhs) const;
private:
void const* m_p;
};
const_iterator begin() const;
const_iterator end() const;
private:
StringTable_hdr const* m_p;
void const* m_strings;
};
/** \brief Represents a variable file info part of a version information block
*
* \ingroup group__library__system
*/
class VsVarFileInfo
: public stlsoft_ns_qual(stl_collection_tag)
{
public:
/// This type
typedef VsVarFileInfo class_type;
/// The value type
typedef VsVar value_type;
public:
/// Constructor
///
/// \param p The header of the block for which the instance will act
VsVarFileInfo(VarFileInfo_hdr const* p);
/// The Key property
wchar_t const* Key() const;
/// Iterator class
class const_iterator
: public stlsoft_ns_qual(iterator_base)<winstl_ns_qual_std(forward_iterator_tag)
, value_type
, ws_ptrdiff_t
, void // By-Value Temporary reference
, value_type // By-Value Temporary reference
>
{
public:
/// This type
typedef const_iterator class_type;
public:
/// Constructor
const_iterator(void const* p);
/// Pre-increment operator
class_type& operator ++();
/// Post-increment operator
class_type operator ++(int);
value_type operator *() const;
ws_bool_t operator ==(class_type const& rhs) const;
ws_bool_t operator !=(class_type const& rhs) const;
private:
void const* m_p;
};
const_iterator begin() const;
const_iterator end() const;
private:
VarFileInfo_hdr const* m_p;
void const* m_vars;
};
/** \brief Represents a variable string part of a version information block
*
* \ingroup group__library__system
*/
class VsStringFileInfo
: public stlsoft_ns_qual(stl_collection_tag)
{
public:
/// This type
typedef VsStringFileInfo class_type;
/// The value type
typedef VsStringTable value_type;
public:
/// Constructor
VsStringFileInfo(StringFileInfo_hdr const* p);
/// The key of the StringFileInfo block
wchar_t const* Key() const;
/// Non-mutating (const) iterator type for the StringFileInfo block
///
/// \note The value type is \c VsStringTable
class const_iterator
: public stlsoft_ns_qual(iterator_base)<winstl_ns_qual_std(forward_iterator_tag)
, value_type
, ws_ptrdiff_t
, void // By-Value Temporary reference
, value_type // By-Value Temporary reference
>
{
public:
/// The class type
typedef const_iterator class_type;
/// The value type
typedef VsStringTable value_type;
public:
/// Constructor
const_iterator(void const* p);
/// Pre-increment operator
class_type& operator ++();
/// Post-increment operator
class_type operator ++(int);
value_type operator *() const;
ws_bool_t operator ==(class_type const& rhs) const;
ws_bool_t operator !=(class_type const& rhs) const;
private:
void const* m_p;
};
const_iterator begin() const;
const_iterator end() const;
private:
StringFileInfo_hdr const* m_p;
void const* m_vars;
};
/** \brief Provides convenient access to aspects of a module's version information
*
* \ingroup group__library__system
*/
class version_info
{
private:
typedef processheap_allocator<ws_byte_t> allocator_type;
public:
/// This type
typedef version_info class_type;
/// \name Construction
/// @{
public:
/// Creates an instance corresponding to the version information from the given module
///
/// \param moduleName The name of the module (.exe, .dll, etc.) to load
ss_explicit_k version_info(ws_char_a_t const* moduleName);
/// Creates an instance corresponding to the version information from the given module
///
/// \param moduleName The name of the module (.exe, .dll, etc.) to load
ss_explicit_k version_info(ws_char_w_t const* moduleName);
/// Releases any allocated resources
~version_info() stlsoft_throw_0();
/// @}
/// \name Properties
/// @{
public:
/// The length of the version information
ws_size_t Length() const;
/// The length of the value part of the version block
ws_size_t ValueLength() const;
/// The type field in the version block
ws_size_t Type() const;
/// The key of the version block
wchar_t const* Key() const;
/// The FixedFileInfo part of the block
fixed_file_info FixedFileInfo() const;
/// Indicates whether the module contains a VarFileInfo block
ws_bool_t HasVarFileInfo() const;
/// The VarFileInfo part of the block
VsVarFileInfo VarFileInfo() const;
/// Indicates whether the module contains a StringFileInfo block
ws_bool_t HasStringFileInfo() const;
/// The StringFileInfo part of the block
VsStringFileInfo StringFileInfo() const;
/// @}
private:
static VS_VERSIONINFO_hdr const* retrieve_module_info_block_(ws_char_a_t const* moduleName);
static VS_VERSIONINFO_hdr const* retrieve_module_info_block_(ws_char_w_t const* moduleName);
static wchar_t const* calc_key_(void const* pv);
static VS_FIXEDFILEINFO const* calc_ffi_(wchar_t const* key);
static WORD const* calc_children_(VS_FIXEDFILEINFO const* ffi);
private:
void init_();
private:
VS_VERSIONINFO_hdr const* const m_hdr;
wchar_t const* const m_key;
VS_FIXEDFILEINFO const* const m_ffi;
WORD const* const m_children;
StringFileInfo_hdr const* m_sfi;
VarFileInfo_hdr const* m_vfi;
// Not to be implemented
private:
version_info(class_type const& rhs);
class_type& operator =(class_type const& rhs);
};
////////////////////////////////////////////////////////////////////////////
// Unit-testing
#ifdef STLSOFT_UNITTEST
# include "./unittest/version_info_unittest_.h"
#endif /* STLSOFT_UNITTEST */
/* /////////////////////////////////////////////////////////////////////////
* Implementation
*/
#ifndef STLSOFT_DOCUMENTATION_SKIP_SECTION
inline /* static */ FILETIME fixed_file_info::calc_FileDateTime_(VS_FIXEDFILEINFO const* ffi)
{
FILETIME ft = { ffi->dwFileDateLS, ffi->dwFileDateMS };
return ft;
}
inline fixed_file_info::fixed_file_info(VS_FIXEDFILEINFO const* ffi)
: m_ffi(ffi)
, m_fileDateTime(calc_FileDateTime_(ffi))
{}
inline ws_uint16_t fixed_file_info::ApiVerHigh() const
{
return HIWORD(m_ffi->dwStrucVersion);
}
inline ws_uint16_t fixed_file_info::ApiVerLow() const
{
return LOWORD(m_ffi->dwStrucVersion);
}
inline ws_uint16_t fixed_file_info::FileVerMajor() const
{
return HIWORD(m_ffi->dwFileVersionMS);
}
inline ws_uint16_t fixed_file_info::FileVerMinor() const
{
return LOWORD(m_ffi->dwFileVersionMS);
}
inline ws_uint16_t fixed_file_info::FileVerRevision() const
{
return HIWORD(m_ffi->dwFileVersionLS);
}
inline ws_uint16_t fixed_file_info::FileVerBuild() const
{
return LOWORD(m_ffi->dwFileVersionLS);
}
inline ws_uint16_t fixed_file_info::ProductVerMajor() const
{
return HIWORD(m_ffi->dwProductVersionMS);
}
inline ws_uint16_t fixed_file_info::ProductVerMinor() const
{
return LOWORD(m_ffi->dwProductVersionMS);
}
inline ws_uint16_t fixed_file_info::ProductVerRevision() const
{
return HIWORD(m_ffi->dwProductVersionLS);
}
inline ws_uint16_t fixed_file_info::ProductVerBuild() const
{
return LOWORD(m_ffi->dwProductVersionLS);
}
inline ws_uint32_t fixed_file_info::FileFlagsMask() const
{
return m_ffi->dwFileFlagsMask;
}
inline ws_uint32_t fixed_file_info::FileFlags() const
{
return m_ffi->dwFileFlags;
}
inline ws_uint32_t fixed_file_info::FileOS() const
{
return m_ffi->dwFileOS;
}
inline ws_uint32_t fixed_file_info::FileType() const
{
return m_ffi->dwFileType;
}
inline ws_uint32_t fixed_file_info::FileSubtype() const
{
return m_ffi->dwFileSubtype;
}
inline FILETIME const& fixed_file_info::FileDateTime() const
{
return m_fileDateTime;
}
inline VsVar::VsVar(Var_hdr const* p)
: m_p(p)
{
WINSTL_ASSERT(0 == ::wcsncmp(p->szKey, L"Translation", 12));
m_values = sap_cast<LangCodePage const*>(rounded_ptr(&p->szKey[1 + ::wcslen(p->szKey)], 4));
}
inline ss_size_t VsVar::length() const
{
return m_p->wValueLength / sizeof(LangCodePage);
}
inline VsVar::LangCodePage const& VsVar::operator [](ss_size_t index) const
{
return m_values[index];
}
inline VsString::VsString(String_hdr const* p)
: m_name(p->szKey)
{
m_value = sap_cast<wchar_t const*>(rounded_ptr(&p->szKey[1 + ::wcslen(p->szKey)], 4));
}
inline wchar_t const* VsString::name() const
{
return m_name;
}
inline wchar_t const* VsString::value() const
{
return m_value;
}
inline VsStringTable::VsStringTable(StringTable_hdr const* p)
: m_p(p)
{
m_strings = rounded_ptr(&p->szKey[1 + ::wcslen(p->szKey)], 4);
}
inline wchar_t const* VsStringTable::Key() const
{
WINSTL_ASSERT(NULL != m_p);
return m_p->szKey;
}
inline VsStringTable::const_iterator::const_iterator(void const* p)
: m_p(p)
{}
inline VsStringTable::const_iterator::class_type& VsStringTable::const_iterator::operator ++()
{
String_hdr const* str = static_cast<String_hdr const*>(m_p);
m_p = rounded_ptr(m_p, str->wLength, 4);
return *this;
}
inline VsStringTable::const_iterator::class_type VsStringTable::const_iterator::operator ++(int)
{
const_iterator ret(*this);
operator ++();
return ret;
}
inline VsString VsStringTable::const_iterator::operator *() const
{
String_hdr const* str = static_cast<String_hdr const*>(m_p);
return VsString(str);
}
inline ws_bool_t VsStringTable::const_iterator::operator ==(VsStringTable::const_iterator::class_type const& rhs) const
{
return m_p == rhs.m_p;
}
inline ws_bool_t VsStringTable::const_iterator::operator !=(VsStringTable::const_iterator::class_type const& rhs) const
{
return !operator ==(rhs);
}
inline VsStringTable::const_iterator VsStringTable::begin() const
{
return const_iterator(m_strings);
}
inline VsStringTable::const_iterator VsStringTable::end() const
{
return const_iterator(rounded_ptr(m_p, m_p->wLength, 4));
}
inline VsVarFileInfo::VsVarFileInfo(VarFileInfo_hdr const* p)
: m_p(p)
{
WINSTL_ASSERT(0 == ::wcsncmp(p->szKey, L"VarFileInfo", 12));
m_vars = rounded_ptr(&p->szKey[1 + ::wcslen(p->szKey)], 4);
}
inline wchar_t const* VsVarFileInfo::Key() const
{
WINSTL_ASSERT(NULL != m_p);
return m_p->szKey;
}
inline VsVarFileInfo::const_iterator::const_iterator(void const* p)
: m_p(p)
{}
inline VsVarFileInfo::const_iterator::class_type& VsVarFileInfo::const_iterator::operator ++()
{
Var_hdr const* var = static_cast<Var_hdr const*>(m_p);
m_p = rounded_ptr(m_p, var->wLength, 4);
return *this;
}
inline VsVarFileInfo::const_iterator::class_type VsVarFileInfo::const_iterator::operator ++(int)
{
const_iterator ret(*this);
operator ++();
return ret;
}
inline VsVar VsVarFileInfo::const_iterator::operator *() const
{
Var_hdr const* var = static_cast<Var_hdr const*>(m_p);
return VsVar(var);
}
inline ws_bool_t VsVarFileInfo::const_iterator::operator ==(class_type const& rhs) const
{
return m_p == rhs.m_p;
}
inline ws_bool_t VsVarFileInfo::const_iterator::operator !=(class_type const& rhs) const
{
return !operator ==(rhs);
}
inline VsVarFileInfo::const_iterator VsVarFileInfo::begin() const
{
return const_iterator(m_vars);
}
inline VsVarFileInfo::const_iterator VsVarFileInfo::end() const
{
return const_iterator(rounded_ptr(m_p, m_p->wLength, 4));
}
inline VsStringFileInfo::VsStringFileInfo(StringFileInfo_hdr const* p)
: m_p(p)
{
WINSTL_ASSERT(0 == ::wcsncmp(p->szKey, L"StringFileInfo", 15));
m_vars = rounded_ptr(&p->szKey[1 + ::wcslen(p->szKey)], 4);
}
inline wchar_t const* VsStringFileInfo::Key() const
{
WINSTL_ASSERT(NULL != m_p);
return m_p->szKey;
}
inline VsStringFileInfo::const_iterator::const_iterator(void const* p)
: m_p(p)
{}
inline VsStringFileInfo::const_iterator::class_type& VsStringFileInfo::const_iterator::operator ++()
{
StringTable_hdr const* strtbl = static_cast<StringTable_hdr const*>(m_p);
m_p = rounded_ptr(m_p, strtbl->wLength, 4);
return *this;
}
inline VsStringFileInfo::const_iterator::class_type VsStringFileInfo::const_iterator::operator ++(int)
{
const_iterator ret(*this);
operator ++();
return ret;
}
inline VsStringTable VsStringFileInfo::const_iterator::operator *() const
{
StringTable_hdr const* strtbl = static_cast<StringTable_hdr const*>(m_p);
return VsStringTable(strtbl);
}
inline ws_bool_t VsStringFileInfo::const_iterator::operator ==(class_type const& rhs) const
{
return m_p == rhs.m_p;
}
inline ws_bool_t VsStringFileInfo::const_iterator::operator !=(class_type const& rhs) const
{
return !operator ==(rhs);
}
inline VsStringFileInfo::const_iterator VsStringFileInfo::begin() const
{
return const_iterator(m_vars);
}
inline VsStringFileInfo::const_iterator VsStringFileInfo::end() const
{
return const_iterator(rounded_ptr(m_p, m_p->wLength, 4));
}
inline /* ss_explicit_k */ version_info::version_info(ws_char_a_t const* moduleName)
: m_hdr(retrieve_module_info_block_(moduleName))
, m_key(calc_key_(m_hdr))
, m_ffi(calc_ffi_(m_key))
, m_children(calc_children_(m_ffi))
, m_sfi(NULL)
, m_vfi(NULL)
{
init_();
}
inline /* ss_explicit_k */ version_info::version_info(ws_char_w_t const* moduleName)
: m_hdr(retrieve_module_info_block_(moduleName))
, m_key(calc_key_(m_hdr))
, m_ffi(calc_ffi_(m_key))
, m_children(calc_children_(m_ffi))
, m_sfi(NULL)
, m_vfi(NULL)
{
init_();
}
inline version_info::~version_info() stlsoft_throw_0()
{
allocator_type allocator;
allocator.deallocate(const_cast<ws_byte_t*>(sap_cast<ws_byte_t const*>(m_hdr)));
}
inline ws_size_t version_info::Length() const
{
#if !defined(STLSOFT_CF_EXCEPTION_SUPPORT) || \
!defined(STLSOFT_CF_THROW_BAD_ALLOC)
if(NULL == m_hdr)
{
return 0;
}
#else /* ? exceptions */
WINSTL_ASSERT(NULL != m_hdr);
#endif /* !STLSOFT_CF_EXCEPTION_SUPPORT || !STLSOFT_CF_THROW_BAD_ALLOC */
return *(sap_cast<WORD const*>(m_hdr) + 0);
}
inline ws_size_t version_info::ValueLength() const
{
#if !defined(STLSOFT_CF_EXCEPTION_SUPPORT) || \
!defined(STLSOFT_CF_THROW_BAD_ALLOC)
if(NULL == m_hdr)
{
return 0;
}
#else /* ? exceptions */
WINSTL_ASSERT(NULL != m_hdr);
#endif /* !STLSOFT_CF_EXCEPTION_SUPPORT || !STLSOFT_CF_THROW_BAD_ALLOC */
return *(sap_cast<WORD const*>(m_hdr) + 1);
}
inline ws_size_t version_info::Type() const
{
WINSTL_ASSERT(NULL != m_hdr);
return *(sap_cast<WORD const*>(m_hdr) + 2);
}
inline wchar_t const* version_info::Key() const
{
WINSTL_ASSERT(NULL != m_hdr);
return m_key;
}
inline fixed_file_info version_info::FixedFileInfo() const
{
WINSTL_ASSERT(NULL != m_hdr);
return fixed_file_info(m_ffi);
}
inline ws_bool_t version_info::HasVarFileInfo() const
{
return NULL != m_vfi;
}
inline VsVarFileInfo version_info::VarFileInfo() const
{
WINSTL_ASSERT(NULL != m_vfi);
return VsVarFileInfo(m_vfi);
}
inline ws_bool_t version_info::HasStringFileInfo() const
{
return NULL != m_sfi;
}
inline VsStringFileInfo version_info::StringFileInfo() const
{
WINSTL_ASSERT(NULL != m_sfi);
return VsStringFileInfo(m_sfi);
}
inline /* static */ VS_VERSIONINFO_hdr const* version_info::retrieve_module_info_block_(ws_char_a_t const* moduleName)
{
#ifdef WINSTL_VERSION_INFO_NO_USE_FILE_PATH_BUFFER
ws_char_a_t buffer[1 + WINSTL_CONST_MAX_PATH];
#else /* ?WINSTL_VERSION_INFO_NO_USE_FILE_PATH_BUFFER */
basic_file_path_buffer<ws_char_a_t> buffer;
#endif /* WINSTL_VERSION_INFO_NO_USE_FILE_PATH_BUFFER */
if( NULL == moduleName &&
#ifdef WINSTL_VERSION_INFO_NO_USE_FILE_PATH_BUFFER
0 != ::GetModuleFileNameA(NULL, &buffer[0], STLSOFT_NUM_ELEMENTS(buffer)))
#else /* ?WINSTL_VERSION_INFO_NO_USE_FILE_PATH_BUFFER */
0 != ::GetModuleFileNameA(NULL, &buffer[0], DWORD(buffer.size())))
#endif /* WINSTL_VERSION_INFO_NO_USE_FILE_PATH_BUFFER */
{
moduleName = stlsoft_ns_qual(c_str_ptr)(buffer);
}
else
{
// Must verify it can be loaded, i.e. is a 32-bit resource
//
// TODO: Work out how to support 16-bit versions
HINSTANCE hinst = ::LoadLibraryExA(moduleName, NULL, LOAD_LIBRARY_AS_DATAFILE);
if(NULL == hinst)
{
#ifdef STLSOFT_CF_EXCEPTION_SUPPORT
STLSOFT_THROW_X(version_info_exception("Could not elicit version information from module", ::GetLastError()));
#else /* ? STLSOFT_CF_EXCEPTION_SUPPORT */
return NULL;
#endif /* STLSOFT_CF_EXCEPTION_SUPPORT */
}
else
{
::FreeLibrary(hinst);
}
}
allocator_type allocator;
ws_dword_t cb = ::GetFileVersionInfoSizeA(const_cast<ws_char_a_t*>(moduleName), NULL);
void *pv = (0 == cb) ? NULL : allocator.allocate(cb);
#if !defined(STLSOFT_CF_THROW_BAD_ALLOC)
// If bad_alloc will not be thrown, then we need to check for NULL, but only act on it
// if cb is non-zero
if( 0 != cb &&
pv == NULL)
{
::GetLastError();
return NULL;
}
#endif /* !STLSOFT_CF_THROW_BAD_ALLOC */
WINSTL_ASSERT(0 == cb || pv != NULL);
if( 0 == cb ||
!::GetFileVersionInfoA(const_cast<ws_char_a_t*>(moduleName), 0, cb, pv))
{
allocator.deallocate(static_cast<ws_byte_t*>(pv), cb);
#ifdef STLSOFT_CF_EXCEPTION_SUPPORT
STLSOFT_THROW_X(version_info_exception("Could not elicit version information from module", ::GetLastError()));
#else /* ? STLSOFT_CF_EXCEPTION_SUPPORT */
return NULL;
#endif /* STLSOFT_CF_EXCEPTION_SUPPORT */
}
WINSTL_ASSERT(pv != NULL);
return static_cast<VS_VERSIONINFO_hdr*>(pv);
}
inline /* static */ VS_VERSIONINFO_hdr const* version_info::retrieve_module_info_block_(ws_char_w_t const* moduleName)
{
#ifdef WINSTL_VERSION_INFO_NO_USE_FILE_PATH_BUFFER
ws_char_w_t buffer[1 + WINSTL_CONST_MAX_PATH];
#else /* ?WINSTL_VERSION_INFO_NO_USE_FILE_PATH_BUFFER */
basic_file_path_buffer<ws_char_w_t> buffer;
#endif /* WINSTL_VERSION_INFO_NO_USE_FILE_PATH_BUFFER */
if( NULL == moduleName &&
#ifdef WINSTL_VERSION_INFO_NO_USE_FILE_PATH_BUFFER
0 != ::GetModuleFileNameW(NULL, &buffer[0], STLSOFT_NUM_ELEMENTS(buffer)))
#else /* ?WINSTL_VERSION_INFO_NO_USE_FILE_PATH_BUFFER */
0 != ::GetModuleFileNameW(NULL, &buffer[0], DWORD(buffer.size())))
#endif /* WINSTL_VERSION_INFO_NO_USE_FILE_PATH_BUFFER */
{
moduleName = stlsoft_ns_qual(c_str_ptr)(buffer);
}
else
{
// Must verify it can be loaded, i.e. is a 32-bit resource
//
// TODO: Work out how to support 16-bit versions
HINSTANCE hinst = ::LoadLibraryExW(moduleName, NULL, LOAD_LIBRARY_AS_DATAFILE);
if(NULL == hinst)
{
#ifdef STLSOFT_CF_EXCEPTION_SUPPORT
STLSOFT_THROW_X(version_info_exception("Could not elicit version information from module", ::GetLastError()));
#else /* ? STLSOFT_CF_EXCEPTION_SUPPORT */
return NULL;
#endif /* STLSOFT_CF_EXCEPTION_SUPPORT */
}
else
{
::FreeLibrary(hinst);
}
}
allocator_type allocator;
ws_dword_t cb = ::GetFileVersionInfoSizeW(const_cast<ws_char_w_t*>(moduleName), NULL);
void *pv = (0 == cb) ? NULL : allocator.allocate(cb);
#ifndef STLSOFT_CF_THROW_BAD_ALLOC
if( 0 != cb &&
pv == NULL)
{
return NULL;
}
#endif /* !STLSOFT_CF_THROW_BAD_ALLOC */
if( 0 == cb ||
!::GetFileVersionInfoW(const_cast<ws_char_w_t*>(moduleName), 0, cb, pv))
{
allocator.deallocate(static_cast<ws_byte_t*>(pv), cb);
#ifdef STLSOFT_CF_EXCEPTION_SUPPORT
STLSOFT_THROW_X((version_info_exception("Could not elicit version information from module", ::GetLastError())));
#else /* ? STLSOFT_CF_EXCEPTION_SUPPORT */
pv = NULL;
#endif /* STLSOFT_CF_EXCEPTION_SUPPORT */
}
return static_cast<VS_VERSIONINFO_hdr*>(pv);
}
inline /* static */ wchar_t const* version_info::calc_key_(void const* pv)
{
#if !defined(STLSOFT_CF_EXCEPTION_SUPPORT) || \
!defined(STLSOFT_CF_THROW_BAD_ALLOC)
if(NULL == pv)
{
return NULL;
}
#else /* ? exceptions */
WINSTL_ASSERT(NULL != pv);
#endif /* !STLSOFT_CF_EXCEPTION_SUPPORT || !STLSOFT_CF_THROW_BAD_ALLOC */
#ifdef _DEBUG
// Bit of 16-bit resource code here
//
// This is reasonably safe, because if it is unicode, then the n-limited string comparison
// will simply return non-0, rather than potentially going off and crashing
{
char const* keyA = reinterpret_cast<char const*>(static_cast<WORD const*>(pv) + 2);
if(0 == ::strncmp("VS_VERSION_INFO", keyA, 16))
{
keyA = NULL;
}
}
#endif /* _DEBUG */
wchar_t const* key = reinterpret_cast<wchar_t const*>(static_cast<WORD const*>(pv) + 3);
WINSTL_ASSERT(0 == ::wcsncmp(L"VS_VERSION_INFO", key, 16));
return key;
}
inline /* static */ VS_FIXEDFILEINFO const* version_info::calc_ffi_(wchar_t const* key)
{
#if !defined(STLSOFT_CF_EXCEPTION_SUPPORT) || \
!defined(STLSOFT_CF_THROW_BAD_ALLOC)
if(NULL == key)
{
return NULL;
}
#else /* ? exceptions */
WINSTL_ASSERT(NULL != key);
#endif /* !STLSOFT_CF_EXCEPTION_SUPPORT || !STLSOFT_CF_THROW_BAD_ALLOC */
return sap_cast<VS_FIXEDFILEINFO const*>(rounded_ptr(&key[1 + ::wcslen(key)], 4));
}
inline /* static */ WORD const* version_info::calc_children_(VS_FIXEDFILEINFO const* ffi)
{
#if !defined(STLSOFT_CF_EXCEPTION_SUPPORT) || \
!defined(STLSOFT_CF_THROW_BAD_ALLOC)
if(NULL == ffi)
{
return NULL;
}
#else /* ? exceptions */
WINSTL_ASSERT(NULL != ffi);
#endif /* !STLSOFT_CF_EXCEPTION_SUPPORT || !STLSOFT_CF_THROW_BAD_ALLOC */
return sap_cast<WORD const*>(rounded_ptr(&ffi[1], 4));
}
inline void version_info::init_()
{
#if !defined(STLSOFT_CF_EXCEPTION_SUPPORT) || \
!defined(STLSOFT_CF_THROW_BAD_ALLOC)
if(NULL == m_hdr)
{
return;
}
#else /* ? exceptions */
WINSTL_ASSERT(NULL != m_hdr);
#endif /* !STLSOFT_CF_EXCEPTION_SUPPORT || !STLSOFT_CF_THROW_BAD_ALLOC */
#ifdef _DEBUG
// Check that ffi is the same as the pointer returned from VerQueryValue("\\");
VS_FIXEDFILEINFO *ffi = NULL;
UINT cchInfo = 0;
WINSTL_ASSERT(::VerQueryValueA(const_cast<VS_VERSIONINFO_hdr*>(m_hdr), "\\", reinterpret_cast<void**>(&ffi), &cchInfo));
WINSTL_ASSERT(ffi == m_ffi);
#endif /* _DEBUG */
// Now we must parse the children.
void const * pv = m_children;
void const *const end = rounded_ptr(m_hdr, m_hdr->wLength, 4);
WINSTL_ASSERT(ptr_byte_diff(end, pv) >= 0);
for(; pv != end; )
{
union
{
void const *pv_;
StringFileInfo_hdr const *psfi;
VarFileInfo_hdr const *pvfi;
} u;
u.pv_ = pv;
WINSTL_ASSERT(ptr_byte_diff(pv, m_hdr) < m_hdr->wLength);
if(0 == ::wcsncmp(u.psfi->szKey, L"StringFileInfo", 15))
{
WINSTL_ASSERT(NULL == m_sfi);
m_sfi = u.psfi;
pv = rounded_ptr(pv, u.psfi->wLength, 4);
}
else if(0 == ::wcsncmp(u.psfi->szKey, L"VarFileInfo", 12))
{
WINSTL_ASSERT(NULL == m_vfi);
m_vfi = u.pvfi;
pv = rounded_ptr(pv, u.pvfi->wLength, 4);
}
else
{
#ifdef STLSOFT_UNITTEST
::wprintf(L"Unexpected contents of VS_VERSIONINFO children. pv: 0x%08x; Key: %.*s\n", pv, 20, u.psfi->szKey);
#endif /* STLSOFT_UNITTEST */
WINSTL_MESSAGE_ASSERT("Unexpected contents of VS_VERSIONINFO children", NULL == m_vfi);
break;
}
WINSTL_ASSERT(ptr_byte_diff(pv, end) <= 0);
}
WINSTL_ASSERT(ptr_byte_diff(pv, m_hdr) == m_hdr->wLength);
#ifdef _DEBUG
fixed_file_info fixedInfo = FixedFileInfo();
ws_uint16_t j = fixedInfo.FileVerMajor();
ws_uint16_t n = fixedInfo.FileVerMinor();
ws_uint16_t r = fixedInfo.FileVerRevision();
ws_uint16_t b = fixedInfo.FileVerBuild();
STLSOFT_SUPPRESS_UNUSED(j);
STLSOFT_SUPPRESS_UNUSED(n);
STLSOFT_SUPPRESS_UNUSED(r);
STLSOFT_SUPPRESS_UNUSED(b);
#endif /* _DEBUG */
}
#endif /* !STLSOFT_DOCUMENTATION_SKIP_SECTION */
/* ////////////////////////////////////////////////////////////////////// */
#ifndef _WINSTL_NO_NAMESPACE
# if defined(_STLSOFT_NO_NAMESPACE) || \
defined(STLSOFT_DOCUMENTATION_SKIP_SECTION)
} // namespace winstl
# else
} // namespace winstl_project
} // namespace stlsoft
# endif /* _STLSOFT_NO_NAMESPACE */
#endif /* !_WINSTL_NO_NAMESPACE */
/* ////////////////////////////////////////////////////////////////////// */
#endif /* WINSTL_INCL_WINSTL_SYSTEM_HPP_VERSION_INFO */
/* ///////////////////////////// end of file //////////////////////////// */