447 lines
19 KiB
C++
447 lines
19 KiB
C++
/*************************************************************************/
|
|
/* Copyright (c) 2015 dx, http://kaimi.ru */
|
|
/* */
|
|
/* Permission is hereby granted, free of charge, to any person */
|
|
/* obtaining a copy of this software and associated documentation */
|
|
/* files (the "Software"), to deal in the Software without */
|
|
/* restriction, including without limitation the rights to use, */
|
|
/* copy, modify, merge, publish, distribute, sublicense, and/or */
|
|
/* sell copies of the Software, and to permit persons to whom the */
|
|
/* Software is furnished to do so, subject to the following conditions: */
|
|
/* The above copyright notice and this permission notice shall be */
|
|
/* included in all copies or substantial portions of the Software. */
|
|
/* */
|
|
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
|
|
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
|
|
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
|
|
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
|
|
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
|
|
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
|
|
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
|
/*************************************************************************/
|
|
#include <algorithm>
|
|
#include <string.h>
|
|
#include "resource_cursor_icon_writer.h"
|
|
|
|
namespace pe_bliss
|
|
{
|
|
using namespace pe_win;
|
|
|
|
resource_cursor_icon_writer::resource_cursor_icon_writer(pe_resource_manager& res)
|
|
:res_(res)
|
|
{}
|
|
|
|
//Add icon helper
|
|
void resource_cursor_icon_writer::add_icon(const std::string& icon_file, const resource_data_info* group_icon_info /* or zero */, resource_directory_entry& new_icon_group_entry, const resource_directory::entry_finder& finder, uint32_t language, icon_place_mode mode, uint32_t codepage, uint32_t timestamp)
|
|
{
|
|
//Check icon for correctness
|
|
if(icon_file.length() < sizeof(ico_header))
|
|
throw pe_exception("Incorrect resource icon", pe_exception::resource_incorrect_icon);
|
|
|
|
const ico_header* icon_header = reinterpret_cast<const ico_header*>(&icon_file[0]);
|
|
|
|
unsigned long size_of_headers = sizeof(ico_header) + icon_header->Count * sizeof(icondirentry);
|
|
if(icon_file.length() < size_of_headers || icon_header->Count == 0)
|
|
throw pe_exception("Incorrect resource icon", pe_exception::resource_incorrect_icon);
|
|
|
|
//Enumerate all icons in file
|
|
for(uint16_t i = 0; i != icon_header->Count; ++i)
|
|
{
|
|
//Check icon entries
|
|
const icondirentry* icon_entry = reinterpret_cast<const icondirentry*>(&icon_file[sizeof(ico_header) + i * sizeof(icondirentry)]);
|
|
if(icon_entry->SizeInBytes == 0
|
|
|| icon_entry->ImageOffset < size_of_headers
|
|
|| !pe_utils::is_sum_safe(icon_entry->ImageOffset, icon_entry->SizeInBytes)
|
|
|| icon_entry->ImageOffset + icon_entry->SizeInBytes > icon_file.length())
|
|
throw pe_exception("Incorrect resource icon", pe_exception::resource_incorrect_icon);
|
|
}
|
|
|
|
std::string icon_group_data;
|
|
ico_header* info = 0;
|
|
|
|
if(group_icon_info)
|
|
{
|
|
//If icon group already exists
|
|
{
|
|
icon_group_data = group_icon_info->get_data();
|
|
codepage = group_icon_info->get_codepage(); //Don't change codepage of icon group entry
|
|
}
|
|
|
|
//Check resource data size
|
|
if(icon_group_data.length() < sizeof(ico_header))
|
|
throw pe_exception("Incorrect resource icon", pe_exception::resource_incorrect_icon);
|
|
|
|
//Get icon header
|
|
info = reinterpret_cast<ico_header*>(&icon_group_data[0]);
|
|
|
|
//Check resource data size
|
|
if(icon_group_data.length() < sizeof(ico_header) + info->Count * sizeof(icon_group))
|
|
throw pe_exception("Incorrect resource icon", pe_exception::resource_incorrect_icon);
|
|
|
|
icon_group_data.resize(sizeof(ico_header) + (info->Count + icon_header->Count) * sizeof(icon_group));
|
|
info = reinterpret_cast<ico_header*>(&icon_group_data[0]); //In case if memory was reallocated
|
|
}
|
|
else //Entry not found - icon group doesn't exist
|
|
{
|
|
icon_group_data.resize(sizeof(ico_header) + icon_header->Count * sizeof(icon_group));
|
|
memcpy(&icon_group_data[0], icon_header, sizeof(ico_header));
|
|
}
|
|
|
|
//Search for available icon IDs
|
|
std::vector<uint16_t> icon_id_list(get_icon_or_cursor_free_id_list(pe_resource_viewer::resource_icon, mode, icon_header->Count));
|
|
|
|
//Enumerate all icons in file
|
|
for(uint16_t i = 0; i != icon_header->Count; ++i)
|
|
{
|
|
const icondirentry* icon_entry = reinterpret_cast<const icondirentry*>(&icon_file[sizeof(ico_header) + i * sizeof(icondirentry)]);
|
|
icon_group group = {0};
|
|
|
|
//Fill icon resource header
|
|
group.BitCount = icon_entry->BitCount;
|
|
group.ColorCount = icon_entry->ColorCount;
|
|
group.Height = icon_entry->Height;
|
|
group.Planes = icon_entry->Planes;
|
|
group.Reserved = icon_entry->Reserved;
|
|
group.SizeInBytes = icon_entry->SizeInBytes;
|
|
group.Width = icon_entry->Width;
|
|
group.Number = icon_id_list.at(i);
|
|
|
|
memcpy(&icon_group_data[sizeof(ico_header) + ((info ? info->Count : 0) + i) * sizeof(icon_group)], &group, sizeof(group));
|
|
|
|
//Add icon to resources
|
|
resource_directory_entry new_entry;
|
|
new_entry.set_id(group.Number);
|
|
res_.add_resource(icon_file.substr(icon_entry->ImageOffset, icon_entry->SizeInBytes), pe_resource_viewer::resource_icon, new_entry, resource_directory::entry_finder(group.Number), language, codepage, timestamp);
|
|
}
|
|
|
|
if(info)
|
|
info->Count += icon_header->Count; //Increase icon count, if we're adding icon to existing group
|
|
|
|
{
|
|
//Add or replace icon group data entry
|
|
res_.add_resource(icon_group_data, pe_resource_viewer::resource_icon_group, new_icon_group_entry, finder, language, codepage, timestamp);
|
|
}
|
|
}
|
|
|
|
//Returns free icon or cursor ID list depending on icon_place_mode
|
|
const std::vector<uint16_t> resource_cursor_icon_writer::get_icon_or_cursor_free_id_list(pe_resource_viewer::resource_type type, icon_place_mode mode, uint32_t count)
|
|
{
|
|
//Search for available icon/cursor IDs
|
|
std::vector<uint16_t> icon_cursor_id_list;
|
|
|
|
try
|
|
{
|
|
//If any icon exists
|
|
//List icon IDs
|
|
std::vector<uint32_t> id_list(res_.list_resource_ids(type));
|
|
std::sort(id_list.begin(), id_list.end());
|
|
|
|
//If we are placing icon on free spaces
|
|
//I.e., icon IDs 1, 3, 4, 7, 8 already exist
|
|
//We'll place five icons on IDs 2, 5, 6, 9, 10
|
|
if(mode != icon_place_after_max_icon_id)
|
|
{
|
|
if(!id_list.empty())
|
|
{
|
|
//Determine and list free icon IDs
|
|
for(std::vector<uint32_t>::const_iterator it = id_list.begin(); it != id_list.end(); ++it)
|
|
{
|
|
if(it == id_list.begin())
|
|
{
|
|
if(*it > 1)
|
|
{
|
|
for(uint16_t i = 1; i != *it; ++i)
|
|
{
|
|
icon_cursor_id_list.push_back(i);
|
|
if(icon_cursor_id_list.size() == count)
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else if(*(it - 1) - *it > 1)
|
|
{
|
|
for(uint16_t i = static_cast<uint16_t>(*(it - 1) + 1); i != static_cast<uint16_t>(*it); ++i)
|
|
{
|
|
icon_cursor_id_list.push_back(i);
|
|
if(icon_cursor_id_list.size() == count)
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(icon_cursor_id_list.size() == count)
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
uint32_t max_id = id_list.empty() ? 0 : *std::max_element(id_list.begin(), id_list.end());
|
|
for(uint32_t i = static_cast<uint32_t>(icon_cursor_id_list.size()); i != count; ++i)
|
|
icon_cursor_id_list.push_back(static_cast<uint16_t>(++max_id));
|
|
}
|
|
catch(const pe_exception&) //Entry not found
|
|
{
|
|
for(uint16_t i = 1; i != count + 1; ++i)
|
|
icon_cursor_id_list.push_back(i);
|
|
}
|
|
|
|
return icon_cursor_id_list;
|
|
}
|
|
|
|
//Add cursor helper
|
|
void resource_cursor_icon_writer::add_cursor(const std::string& cursor_file, const resource_data_info* group_cursor_info /* or zero */, resource_directory_entry& new_cursor_group_entry, const resource_directory::entry_finder& finder, uint32_t language, icon_place_mode mode, uint32_t codepage, uint32_t timestamp)
|
|
{
|
|
//Check cursor for correctness
|
|
if(cursor_file.length() < sizeof(cursor_header))
|
|
throw pe_exception("Incorrect resource cursor", pe_exception::resource_incorrect_cursor);
|
|
|
|
const cursor_header* cur_header = reinterpret_cast<const cursor_header*>(&cursor_file[0]);
|
|
|
|
unsigned long size_of_headers = sizeof(cursor_header) + cur_header->Count * sizeof(cursordirentry);
|
|
if(cursor_file.length() < size_of_headers || cur_header->Count == 0)
|
|
throw pe_exception("Incorrect resource cursor", pe_exception::resource_incorrect_cursor);
|
|
|
|
//Enumerate all cursors in file
|
|
for(uint16_t i = 0; i != cur_header->Count; ++i)
|
|
{
|
|
//Check cursor entries
|
|
const cursordirentry* cursor_entry = reinterpret_cast<const cursordirentry*>(&cursor_file[sizeof(cursor_header) + i * sizeof(cursordirentry)]);
|
|
if(cursor_entry->SizeInBytes == 0
|
|
|| cursor_entry->ImageOffset < size_of_headers
|
|
|| !pe_utils::is_sum_safe(cursor_entry->ImageOffset, cursor_entry->SizeInBytes)
|
|
|| cursor_entry->ImageOffset + cursor_entry->SizeInBytes > cursor_file.length())
|
|
throw pe_exception("Incorrect resource cursor", pe_exception::resource_incorrect_cursor);
|
|
}
|
|
|
|
std::string cursor_group_data;
|
|
cursor_header* info = 0;
|
|
|
|
if(group_cursor_info)
|
|
{
|
|
//If cursor group already exists
|
|
{
|
|
cursor_group_data = group_cursor_info->get_data();
|
|
codepage = group_cursor_info->get_codepage(); //Don't change codepage of cursor group entry
|
|
}
|
|
|
|
//Check resource data size
|
|
if(cursor_group_data.length() < sizeof(cursor_header))
|
|
throw pe_exception("Incorrect resource cursor", pe_exception::resource_incorrect_cursor);
|
|
|
|
//Get cursor header
|
|
info = reinterpret_cast<cursor_header*>(&cursor_group_data[0]);
|
|
|
|
//Check resource data size
|
|
if(cursor_group_data.length() < sizeof(cursor_header) + info->Count * sizeof(cursor_group))
|
|
throw pe_exception("Incorrect resource cursor", pe_exception::resource_incorrect_cursor);
|
|
|
|
cursor_group_data.resize(sizeof(cursor_header) + (info->Count + cur_header->Count) * sizeof(cursor_group));
|
|
info = reinterpret_cast<cursor_header*>(&cursor_group_data[0]); //In case if memory was reallocated
|
|
}
|
|
else //Entry not found - cursor group doesn't exist
|
|
{
|
|
cursor_group_data.resize(sizeof(cursor_header) + cur_header->Count * sizeof(cursor_group));
|
|
memcpy(&cursor_group_data[0], cur_header, sizeof(cursor_header));
|
|
}
|
|
|
|
//Search for available cursor IDs
|
|
std::vector<uint16_t> cursor_id_list(get_icon_or_cursor_free_id_list(pe_resource_viewer::resource_cursor, mode, cur_header->Count));
|
|
|
|
//Enumerate all cursors in file
|
|
for(uint16_t i = 0; i != cur_header->Count; ++i)
|
|
{
|
|
const cursordirentry* cursor_entry = reinterpret_cast<const cursordirentry*>(&cursor_file[sizeof(cursor_header) + i * sizeof(cursordirentry)]);
|
|
cursor_group group = {0};
|
|
|
|
//Fill cursor resource header
|
|
group.Height = cursor_entry->Height * 2;
|
|
group.SizeInBytes = cursor_entry->SizeInBytes + 2 * sizeof(uint16_t) /* hotspot coordinates */;
|
|
group.Width = cursor_entry->Width;
|
|
group.Number = cursor_id_list.at(i);
|
|
|
|
memcpy(&cursor_group_data[sizeof(cursor_header) + ((info ? info->Count : 0) + i) * sizeof(cursor_group)], &group, sizeof(group));
|
|
|
|
//Add cursor to resources
|
|
resource_directory_entry new_entry;
|
|
new_entry.set_id(group.Number);
|
|
|
|
//Fill resource data (two WORDs for hotspot of cursor, and cursor bitmap data)
|
|
std::string cur_data;
|
|
cur_data.resize(sizeof(uint16_t) * 2);
|
|
memcpy(&cur_data[0], &cursor_entry->HotspotX, sizeof(uint16_t));
|
|
memcpy(&cur_data[sizeof(uint16_t)], &cursor_entry->HotspotY, sizeof(uint16_t));
|
|
cur_data.append(cursor_file.substr(cursor_entry->ImageOffset, cursor_entry->SizeInBytes));
|
|
|
|
res_.add_resource(cur_data, pe_resource_viewer::resource_cursor, new_entry, resource_directory::entry_finder(group.Number), language, codepage, timestamp);
|
|
}
|
|
|
|
if(info)
|
|
info->Count += cur_header->Count; //Increase cursor count, if we're adding cursor to existing group
|
|
|
|
{
|
|
//Add or replace cursor group data entry
|
|
res_.add_resource(cursor_group_data, pe_resource_viewer::resource_cursor_group, new_cursor_group_entry, finder, language, codepage, timestamp);
|
|
}
|
|
}
|
|
|
|
//Adds icon(s) from icon file data
|
|
//timestamp will be used for directories that will be added
|
|
//If icon group with name "icon_group_name" or ID "icon_group_id" already exists, it will be appended with new icon(s)
|
|
//(Codepage of icon group and icons will not be changed in this case)
|
|
//icon_place_mode determines, how new icon(s) will be placed
|
|
void resource_cursor_icon_writer::add_icon(const std::string& icon_file, const std::wstring& icon_group_name, uint32_t language, icon_place_mode mode, uint32_t codepage, uint32_t timestamp)
|
|
{
|
|
resource_directory_entry new_icon_group_entry;
|
|
new_icon_group_entry.set_name(icon_group_name);
|
|
std::auto_ptr<resource_data_info> data_info;
|
|
|
|
try
|
|
{
|
|
data_info.reset(new resource_data_info(res_.get_resource_data_by_name(language, pe_resource_viewer::resource_icon_group, icon_group_name)));
|
|
}
|
|
catch(const pe_exception&) //Entry not found
|
|
{
|
|
}
|
|
|
|
add_icon(icon_file, data_info.get(), new_icon_group_entry, resource_directory::entry_finder(icon_group_name), language, mode, codepage, timestamp);
|
|
}
|
|
|
|
void resource_cursor_icon_writer::add_icon(const std::string& icon_file, uint32_t icon_group_id, uint32_t language, icon_place_mode mode, uint32_t codepage, uint32_t timestamp)
|
|
{
|
|
resource_directory_entry new_icon_group_entry;
|
|
new_icon_group_entry.set_id(icon_group_id);
|
|
std::auto_ptr<resource_data_info> data_info;
|
|
|
|
try
|
|
{
|
|
data_info.reset(new resource_data_info(res_.get_resource_data_by_id(language, pe_resource_viewer::resource_icon_group, icon_group_id)));
|
|
}
|
|
catch(const pe_exception&) //Entry not found
|
|
{
|
|
}
|
|
|
|
add_icon(icon_file, data_info.get(), new_icon_group_entry, resource_directory::entry_finder(icon_group_id), language, mode, codepage, timestamp);
|
|
}
|
|
|
|
//Adds cursor(s) from cursor file data
|
|
//timestamp will be used for directories that will be added
|
|
//If cursor group with name "cursor_group_name" or ID "cursor_group_id" already exists, it will be appended with new cursor(s)
|
|
//(Codepage of cursor group and cursors will not be changed in this case)
|
|
//icon_place_mode determines, how new cursor(s) will be placed
|
|
void resource_cursor_icon_writer::add_cursor(const std::string& cursor_file, const std::wstring& cursor_group_name, uint32_t language, icon_place_mode mode, uint32_t codepage, uint32_t timestamp)
|
|
{
|
|
resource_directory_entry new_cursor_group_entry;
|
|
new_cursor_group_entry.set_name(cursor_group_name);
|
|
std::auto_ptr<resource_data_info> data_info;
|
|
|
|
try
|
|
{
|
|
data_info.reset(new resource_data_info(res_.get_resource_data_by_name(language, pe_resource_viewer::resource_cursor_group, cursor_group_name)));
|
|
}
|
|
catch(const pe_exception&) //Entry not found
|
|
{
|
|
}
|
|
|
|
add_cursor(cursor_file, data_info.get(), new_cursor_group_entry, resource_directory::entry_finder(cursor_group_name), language, mode, codepage, timestamp);
|
|
}
|
|
|
|
void resource_cursor_icon_writer::add_cursor(const std::string& cursor_file, uint32_t cursor_group_id, uint32_t language, icon_place_mode mode, uint32_t codepage, uint32_t timestamp)
|
|
{
|
|
resource_directory_entry new_cursor_group_entry;
|
|
new_cursor_group_entry.set_id(cursor_group_id);
|
|
std::auto_ptr<resource_data_info> data_info;
|
|
|
|
try
|
|
{
|
|
data_info.reset(new resource_data_info(res_.get_resource_data_by_id(language, pe_resource_viewer::resource_cursor_group, cursor_group_id)));
|
|
}
|
|
catch(const pe_exception&) //Entry not found
|
|
{
|
|
}
|
|
|
|
add_cursor(cursor_file, data_info.get(), new_cursor_group_entry, resource_directory::entry_finder(cursor_group_id), language, mode, codepage, timestamp);
|
|
}
|
|
|
|
//Remove icon group helper
|
|
void resource_cursor_icon_writer::remove_icons_from_icon_group(const std::string& icon_group_data, uint32_t language)
|
|
{
|
|
//Check resource data size
|
|
if(icon_group_data.length() < sizeof(ico_header))
|
|
throw pe_exception("Incorrect resource icon", pe_exception::resource_incorrect_icon);
|
|
|
|
//Get icon header
|
|
const ico_header* info = reinterpret_cast<const ico_header*>(icon_group_data.data());
|
|
|
|
uint16_t icon_count = info->Count;
|
|
|
|
//Check resource data size
|
|
if(icon_group_data.length() < sizeof(ico_header) + icon_count * sizeof(icon_group))
|
|
throw pe_exception("Incorrect resource icon", pe_exception::resource_incorrect_icon);
|
|
|
|
//Remove icon data
|
|
for(uint16_t i = 0; i != icon_count; ++i)
|
|
{
|
|
const icon_group* group = reinterpret_cast<const icon_group*>(icon_group_data.data() + sizeof(ico_header) + i * sizeof(icon_group));
|
|
res_.remove_resource(pe_resource_viewer::resource_icon, group->Number, language);
|
|
}
|
|
}
|
|
|
|
//Remove cursor group helper
|
|
void resource_cursor_icon_writer::remove_cursors_from_cursor_group(const std::string& cursor_group_data, uint32_t language)
|
|
{
|
|
//Check resource data size
|
|
if(cursor_group_data.length() < sizeof(cursor_header))
|
|
throw pe_exception("Incorrect resource cursor", pe_exception::resource_incorrect_cursor);
|
|
|
|
//Get icon header
|
|
const cursor_header* info = reinterpret_cast<const cursor_header*>(cursor_group_data.data());
|
|
|
|
uint16_t cursor_count = info->Count;
|
|
|
|
//Check resource data size
|
|
if(cursor_group_data.length() < sizeof(cursor_header) + cursor_count * sizeof(cursor_group))
|
|
throw pe_exception("Incorrect resource cursor", pe_exception::resource_incorrect_cursor);
|
|
|
|
//Remove icon data
|
|
for(uint16_t i = 0; i != cursor_count; ++i)
|
|
{
|
|
const icon_group* group = reinterpret_cast<const icon_group*>(cursor_group_data.data() + sizeof(cursor_header) + i * sizeof(cursor_group));
|
|
res_.remove_resource(pe_resource_viewer::resource_cursor, group->Number, language);
|
|
}
|
|
}
|
|
|
|
//Removes cursor group and all its cursors by name/ID and language
|
|
bool resource_cursor_icon_writer::remove_cursor_group(const std::wstring& cursor_group_name, uint32_t language)
|
|
{
|
|
//Get resource by name and language
|
|
const std::string data = res_.get_resource_data_by_name(language, pe_resource_viewer::resource_cursor_group, cursor_group_name).get_data();
|
|
remove_cursors_from_cursor_group(data, language);
|
|
return res_.remove_resource(pe_resource_viewer::resource_cursor_group, cursor_group_name, language);
|
|
}
|
|
|
|
//Removes cursor group and all its cursors by name/ID and language
|
|
bool resource_cursor_icon_writer::remove_cursor_group(uint32_t cursor_group_id, uint32_t language)
|
|
{
|
|
//Get resource by name and language
|
|
const std::string data = res_.get_resource_data_by_id(language, pe_resource_viewer::resource_cursor_group, cursor_group_id).get_data();
|
|
remove_cursors_from_cursor_group(data, language);
|
|
return res_.remove_resource(pe_resource_viewer::resource_cursor_group, cursor_group_id, language);
|
|
}
|
|
|
|
//Removes icon group and all its icons by name/ID and language
|
|
bool resource_cursor_icon_writer::remove_icon_group(const std::wstring& icon_group_name, uint32_t language)
|
|
{
|
|
//Get resource by name and language
|
|
const std::string data = res_.get_resource_data_by_name(language, pe_resource_viewer::resource_icon_group, icon_group_name).get_data();
|
|
remove_icons_from_icon_group(data, language);
|
|
return res_.remove_resource(pe_resource_viewer::resource_icon_group, icon_group_name, language);
|
|
}
|
|
|
|
//Removes icon group and all its icons by name/ID and language
|
|
bool resource_cursor_icon_writer::remove_icon_group(uint32_t icon_group_id, uint32_t language)
|
|
{
|
|
//Get resource by name and language
|
|
const std::string data = res_.get_resource_data_by_id(language, pe_resource_viewer::resource_icon_group, icon_group_id).get_data();
|
|
remove_icons_from_icon_group(data, language);
|
|
return res_.remove_resource(pe_resource_viewer::resource_icon_group, icon_group_id, language);
|
|
}
|
|
}
|