rpcsx/ps3fw/cellPhotoImport.cpp

384 lines
11 KiB
C++
Raw Normal View History

2016-03-21 20:43:03 +01:00
#include "Emu/Cell/PPUModule.h"
2018-02-04 12:46:55 +01:00
#include "Emu/Cell/lv2/sys_fs.h"
#include "Emu/IdManager.h"
2022-07-23 21:16:43 +02:00
#include "Emu/RSX/Overlays/overlay_media_list_dialog.h"
#include "Emu/System.h"
#include "Emu/VFS.h"
2018-02-04 12:46:55 +01:00
#include "cellSysutil.h"
#include "stdafx.h"
#include "util/StrUtil.h"
LOG_CHANNEL(cellPhotoImportUtil, "cellPhotoImport");
// Return Codes
2020-07-16 12:14:57 +02:00
enum CellPhotoImportError : u32
{
CELL_PHOTO_IMPORT_ERROR_BUSY = 0x8002c701,
CELL_PHOTO_IMPORT_ERROR_INTERNAL = 0x8002c702,
CELL_PHOTO_IMPORT_ERROR_PARAM = 0x8002c703,
2014-04-04 15:25:38 +02:00
CELL_PHOTO_IMPORT_ERROR_ACCESS_ERROR = 0x8002c704,
CELL_PHOTO_IMPORT_ERROR_COPY = 0x8002c705,
CELL_PHOTO_IMPORT_ERROR_INITIALIZE = 0x8002c706,
};
template <>
2020-07-16 12:14:57 +02:00
void fmt_class_string<CellPhotoImportError>::format(std::string& out, u64 arg)
{
format_enum(out, arg, [](auto error)
{
switch (error)
{
STR_CASE(CELL_PHOTO_IMPORT_ERROR_BUSY);
STR_CASE(CELL_PHOTO_IMPORT_ERROR_INTERNAL);
STR_CASE(CELL_PHOTO_IMPORT_ERROR_PARAM);
STR_CASE(CELL_PHOTO_IMPORT_ERROR_ACCESS_ERROR);
STR_CASE(CELL_PHOTO_IMPORT_ERROR_COPY);
STR_CASE(CELL_PHOTO_IMPORT_ERROR_INITIALIZE);
}
return unknown;
});
2020-07-16 12:14:57 +02:00
}
2022-07-23 21:16:43 +02:00
enum CellPhotoImportVersion : u32
{
CELL_PHOTO_IMPORT_VERSION_CURRENT = 0,
};
2018-02-04 12:46:55 +01:00
enum
2018-02-09 15:49:37 +01:00
{
CELL_PHOTO_IMPORT_HDD_PATH_MAX = 1055,
2018-02-04 12:46:55 +01:00
CELL_PHOTO_IMPORT_PHOTO_TITLE_MAX_LENGTH = 64,
CELL_PHOTO_IMPORT_GAME_TITLE_MAX_SIZE = 128,
CELL_PHOTO_IMPORT_GAME_COMMENT_MAX_SIZE = 1024
2018-02-09 15:49:37 +01:00
};
2018-02-04 12:46:55 +01:00
enum CellPhotoImportFormatType
2018-02-09 15:49:37 +01:00
{
2018-02-04 12:46:55 +01:00
CELL_PHOTO_IMPORT_FT_UNKNOWN = 0,
CELL_PHOTO_IMPORT_FT_JPEG,
CELL_PHOTO_IMPORT_FT_PNG,
CELL_PHOTO_IMPORT_FT_GIF,
CELL_PHOTO_IMPORT_FT_BMP,
CELL_PHOTO_IMPORT_FT_TIFF,
CELL_PHOTO_IMPORT_FT_MPO,
};
2018-02-04 12:46:55 +01:00
enum CellPhotoImportTexRot
2018-02-09 15:49:37 +01:00
{
2018-02-04 12:46:55 +01:00
CELL_PHOTO_IMPORT_TEX_ROT_0 = 0,
CELL_PHOTO_IMPORT_TEX_ROT_90,
CELL_PHOTO_IMPORT_TEX_ROT_180,
CELL_PHOTO_IMPORT_TEX_ROT_270,
};
struct CellPhotoImportFileDataSub
{
2018-02-04 12:46:55 +01:00
be_t<s32> width;
be_t<s32> height;
be_t<CellPhotoImportFormatType> format;
be_t<CellPhotoImportTexRot> rotate;
};
2018-02-04 12:46:55 +01:00
struct CellPhotoImportFileData
{
2018-02-04 12:46:55 +01:00
char dstFileName[CELL_FS_MAX_FS_FILE_NAME_LENGTH];
char photo_title[CELL_PHOTO_IMPORT_PHOTO_TITLE_MAX_LENGTH * 3];
char game_title[CELL_PHOTO_IMPORT_GAME_TITLE_MAX_SIZE];
char game_comment[CELL_PHOTO_IMPORT_GAME_COMMENT_MAX_SIZE];
char padding;
vm::bptr<CellPhotoImportFileDataSub> data_sub;
vm::bptr<void> reserved;
};
2018-02-04 12:46:55 +01:00
struct CellPhotoImportSetParam
{
2018-02-04 12:46:55 +01:00
be_t<u32> fileSizeMax;
vm::bptr<void> reserved1;
vm::bptr<void> reserved2;
};
using CellPhotoImportFinishCallback =
void(s32 result, vm::ptr<CellPhotoImportFileData> filedata,
vm::ptr<void> userdata);
2018-02-04 12:46:55 +01:00
2022-07-23 21:16:43 +02:00
struct photo_import
{
shared_mutex mutex;
bool is_busy = false;
CellPhotoImportSetParam param{};
vm::ptr<CellPhotoImportFinishCallback> func_finish{};
vm::ptr<void> userdata{};
};
vm::gvar<CellPhotoImportFileDataSub> g_filedata_sub;
vm::gvar<CellPhotoImportFileData> g_filedata;
2022-07-23 21:16:43 +02:00
error_code select_photo(std::string dst_dir)
{
auto& pi_manager = g_fxo->get<photo_import>();
if (!pi_manager.func_finish)
{
cellPhotoImportUtil.error("func_finish is null");
2022-07-23 21:16:43 +02:00
return CELL_PHOTO_IMPORT_ERROR_PARAM;
}
if (!dst_dir.starts_with("/dev_hdd0"sv) &&
!dst_dir.starts_with("/dev_hdd1"sv))
{
cellPhotoImportUtil.error(
"Destination '%s' is not inside dev_hdd0 or dev_hdd1", dst_dir);
return CELL_PHOTO_IMPORT_ERROR_ACCESS_ERROR; // TODO: is this correct?
}
dst_dir = vfs::get(dst_dir);
2022-07-23 21:16:43 +02:00
if (!fs::is_dir(dst_dir))
{
// TODO: check if the dir is user accessible and can be written to
cellPhotoImportUtil.error("Destination '%s' is not a directory", dst_dir);
return CELL_PHOTO_IMPORT_ERROR_ACCESS_ERROR; // TODO: is this correct?
2022-07-23 21:16:43 +02:00
}
pi_manager.is_busy = true;
const std::string vfs_dir_path = vfs::get("/dev_hdd0/photo");
const std::string title = get_localized_string(
localized_string_id::RSX_OVERLAYS_MEDIA_DIALOG_TITLE_PHOTO_IMPORT);
2022-07-23 21:16:43 +02:00
error_code error = rsx::overlays::show_media_list_dialog(
rsx::overlays::media_list_dialog::media_type::photo, vfs_dir_path, title,
2022-07-23 21:16:43 +02:00
[&pi_manager, dst_dir](s32 status, utils::media_info info)
{
sysutil_register_cb([&pi_manager, dst_dir, info,
status](ppu_thread& ppu) -> s32
2022-07-23 21:16:43 +02:00
{
*g_filedata_sub = {};
*g_filedata = {};
u32 result = status >= 0 ? u32{CELL_OK} : u32{CELL_CANCEL};
if (result == CELL_OK)
2022-07-23 21:16:43 +02:00
{
fs::stat_t f_info{};
if (!fs::get_stat(info.path, f_info) || f_info.is_directory)
{
cellPhotoImportUtil.error(
"Path does not belong to a valid file: '%s'", info.path);
result = CELL_PHOTO_IMPORT_ERROR_ACCESS_ERROR; // TODO: is this
// correct ?
pi_manager.is_busy = false;
pi_manager.func_finish(ppu, result, g_filedata,
pi_manager.userdata);
return CELL_OK;
}
if (f_info.size > pi_manager.param.fileSizeMax)
{
cellPhotoImportUtil.error(
"File size is too large: %d (fileSizeMax=%d)", f_info.size,
pi_manager.param.fileSizeMax);
result = CELL_PHOTO_IMPORT_ERROR_COPY; // TODO: is this correct ?
pi_manager.is_busy = false;
pi_manager.func_finish(ppu, result, g_filedata,
pi_manager.userdata);
return CELL_OK;
}
const std::string filename =
info.path.substr(info.path.find_last_of(fs::delim) + 1);
const std::string title = info.get_metadata("title", filename);
const std::string dst_path = dst_dir + "/" + filename;
std::string sub_type = info.sub_type;
strcpy_trunc(g_filedata->dstFileName, filename);
strcpy_trunc(g_filedata->photo_title, title);
strcpy_trunc(g_filedata->game_title, Emu.GetTitle());
strcpy_trunc(g_filedata->game_comment, ""); // TODO
g_filedata->data_sub = g_filedata_sub;
g_filedata->data_sub->width = info.width;
g_filedata->data_sub->height = info.height;
cellPhotoImportUtil.notice(
"Raw image data: filename='%s', title='%s', game='%s', "
"sub_type='%s', width=%d, height=%d, orientation=%d ",
filename, title, Emu.GetTitle(), sub_type, info.width,
info.height, info.orientation);
// Fallback to extension if necessary
if (sub_type.empty())
{
sub_type = get_file_extension(filename);
}
if (!sub_type.empty())
{
sub_type = fmt::to_lower(sub_type);
}
if (sub_type == "jpg" || sub_type == "jpeg")
{
g_filedata->data_sub->format = CELL_PHOTO_IMPORT_FT_JPEG;
}
else if (sub_type == "png")
{
g_filedata->data_sub->format = CELL_PHOTO_IMPORT_FT_PNG;
}
else if (sub_type == "tif" || sub_type == "tiff")
{
g_filedata->data_sub->format = CELL_PHOTO_IMPORT_FT_TIFF;
}
else if (sub_type == "bmp")
{
g_filedata->data_sub->format = CELL_PHOTO_IMPORT_FT_BMP;
}
else if (sub_type == "gif")
{
g_filedata->data_sub->format = CELL_PHOTO_IMPORT_FT_GIF;
}
else if (sub_type == "mpo")
{
g_filedata->data_sub->format = CELL_PHOTO_IMPORT_FT_MPO;
}
else
{
g_filedata->data_sub->format = CELL_PHOTO_IMPORT_FT_UNKNOWN;
}
switch (info.orientation)
{
default:
case CELL_SEARCH_ORIENTATION_UNKNOWN:
case CELL_SEARCH_ORIENTATION_TOP_LEFT:
g_filedata->data_sub->rotate = CELL_PHOTO_IMPORT_TEX_ROT_0;
break;
case CELL_SEARCH_ORIENTATION_TOP_RIGHT:
g_filedata->data_sub->rotate = CELL_PHOTO_IMPORT_TEX_ROT_90;
break;
case CELL_SEARCH_ORIENTATION_BOTTOM_RIGHT:
g_filedata->data_sub->rotate = CELL_PHOTO_IMPORT_TEX_ROT_180;
break;
case CELL_SEARCH_ORIENTATION_BOTTOM_LEFT:
g_filedata->data_sub->rotate = CELL_PHOTO_IMPORT_TEX_ROT_270;
break;
}
cellPhotoImportUtil.notice(
"Media list dialog: Copying '%s' to '%s'...", info.path,
dst_path);
if (!fs::copy_file(info.path, dst_path, false))
{
cellPhotoImportUtil.error(
"Failed to copy '%s' to '%s'. Error = '%s'", info.path,
dst_path, fs::g_tls_error);
result = CELL_PHOTO_IMPORT_ERROR_COPY;
}
cellPhotoImportUtil.notice(
"Cell image data: dstFileName='%s', photo_title='%s', "
"game_title='%s', format=%d, width=%d, height=%d, rotate=%d ",
g_filedata->dstFileName, g_filedata->photo_title,
g_filedata->game_title,
static_cast<s32>(g_filedata->data_sub->format),
g_filedata->data_sub->width, g_filedata->data_sub->height,
static_cast<s32>(g_filedata->data_sub->rotate));
2022-07-23 21:16:43 +02:00
}
else
{
cellPhotoImportUtil.notice("Media list dialog was canceled");
2022-07-23 21:16:43 +02:00
}
pi_manager.is_busy = false;
pi_manager.func_finish(ppu, result, g_filedata, pi_manager.userdata);
return CELL_OK;
});
2022-07-23 21:16:43 +02:00
});
if (error != CELL_OK)
{
pi_manager.is_busy = false;
}
return error;
}
error_code cellPhotoImport(u32 version, vm::cptr<char> dstHddPath,
vm::ptr<CellPhotoImportSetParam> param,
u32 container,
vm::ptr<CellPhotoImportFinishCallback> funcFinish,
vm::ptr<void> userdata)
{
cellPhotoImportUtil.todo(
"cellPhotoImport(version=0x%x, dstHddPath=%s, param=*0x%x, "
"container=0x%x, funcFinish=*0x%x, userdata=*0x%x)",
version, dstHddPath, param, container, funcFinish, userdata);
2018-02-04 12:46:55 +01:00
if (version != CELL_PHOTO_IMPORT_VERSION_CURRENT || !funcFinish || !param ||
!dstHddPath)
2018-02-04 12:46:55 +01:00
{
2022-07-23 21:16:43 +02:00
return CELL_PHOTO_IMPORT_ERROR_PARAM;
}
2018-02-04 12:46:55 +01:00
2022-07-23 21:16:43 +02:00
if (container != 0xffffffff && false) // TODO
{
return CELL_PHOTO_IMPORT_ERROR_PARAM;
}
auto& pi_manager = g_fxo->get<photo_import>();
std::lock_guard lock(pi_manager.mutex);
if (pi_manager.is_busy)
{
return CELL_PHOTO_IMPORT_ERROR_BUSY;
}
pi_manager.param = *param;
pi_manager.func_finish = funcFinish;
pi_manager.userdata = userdata;
return select_photo(dstHddPath.get_ptr());
}
error_code cellPhotoImport2(u32 version, vm::cptr<char> dstHddPath,
vm::ptr<CellPhotoImportSetParam> param,
vm::ptr<CellPhotoImportFinishCallback> funcFinish,
vm::ptr<void> userdata)
{
cellPhotoImportUtil.todo("cellPhotoImport2(version=0x%x, dstHddPath=%s, "
"param=*0x%x, funcFinish=*0x%x, userdata=*0x%x)",
version, dstHddPath, param, funcFinish, userdata);
2018-02-04 12:46:55 +01:00
if (version != CELL_PHOTO_IMPORT_VERSION_CURRENT || !funcFinish || !param ||
!dstHddPath)
2018-02-04 12:46:55 +01:00
{
2022-07-23 21:16:43 +02:00
return CELL_PHOTO_IMPORT_ERROR_PARAM;
}
auto& pi_manager = g_fxo->get<photo_import>();
std::lock_guard lock(pi_manager.mutex);
if (pi_manager.is_busy)
{
return CELL_PHOTO_IMPORT_ERROR_BUSY;
}
pi_manager.param = *param;
pi_manager.func_finish = funcFinish;
pi_manager.userdata = userdata;
2018-02-04 12:46:55 +01:00
2022-07-23 21:16:43 +02:00
return select_photo(dstHddPath.get_ptr());
}
2016-03-21 20:43:03 +01:00
DECLARE(ppu_module_manager::cellPhotoImportUtil)("cellPhotoImportUtil", []()
{
REG_FUNC(cellPhotoImportUtil, cellPhotoImport);
REG_FUNC(cellPhotoImportUtil, cellPhotoImport2);
REG_VAR(cellPhotoImportUtil, g_filedata_sub).flag(MFF_HIDDEN);
REG_VAR(cellPhotoImportUtil, g_filedata).flag(MFF_HIDDEN);
});