rpcsx/rpcs3/Emu/Cell/Modules/cellAvconfExt.cpp

336 lines
12 KiB
C++
Raw Normal View History

2018-12-20 23:36:56 +01:00
#include "stdafx.h"
2016-03-21 20:43:03 +01:00
#include "Emu/System.h"
#include "Emu/Cell/PPUModule.h"
#include "Emu/IdManager.h"
#include "Emu/RSX/rsx_utils.h"
2018-12-20 23:36:56 +01:00
#include "Utilities/StrUtil.h"
#include "cellAudioIn.h"
#include "cellAudioOut.h"
#include "cellVideoOut.h"
#include "cellSysutil.h"
LOG_CHANNEL(cellAvconfExt);
2019-11-15 20:15:00 +01:00
template<>
void fmt_class_string<CellAudioInError>::format(std::string& out, u64 arg)
{
format_enum(out, arg, [](auto error)
{
switch (error)
{
STR_CASE(CELL_AUDIO_IN_ERROR_NOT_IMPLEMENTED);
STR_CASE(CELL_AUDIO_IN_ERROR_ILLEGAL_CONFIGURATION);
STR_CASE(CELL_AUDIO_IN_ERROR_ILLEGAL_PARAMETER);
STR_CASE(CELL_AUDIO_IN_ERROR_PARAMETER_OUT_OF_RANGE);
STR_CASE(CELL_AUDIO_IN_ERROR_DEVICE_NOT_FOUND);
STR_CASE(CELL_AUDIO_IN_ERROR_UNSUPPORTED_AUDIO_IN);
STR_CASE(CELL_AUDIO_IN_ERROR_UNSUPPORTED_SOUND_MODE);
STR_CASE(CELL_AUDIO_IN_ERROR_CONDITION_BUSY);
}
return unknown;
});
}
2018-12-20 23:36:56 +01:00
struct avconf_manager
{
std::vector<CellAudioInDeviceInfo> devices;
void copy_device_info(u32 num, vm::ptr<CellAudioInDeviceInfo> info);
avconf_manager();
};
avconf_manager::avconf_manager()
{
u32 curindex = 0;
auto mic_list = fmt::split(g_cfg.audio.microphone_devices, {"@@@"});
if (mic_list.size())
{
switch (g_cfg.audio.microphone_type)
{
case microphone_handler::standard:
for (u32 index = 0; index < mic_list.size(); index++)
{
devices.emplace_back();
devices[curindex].portType = CELL_AUDIO_IN_PORT_USB;
devices[curindex].availableModeCount = 1;
devices[curindex].state = CELL_AUDIO_IN_DEVICE_STATE_AVAILABLE;
devices[curindex].deviceId = 0xE11CC0DE + curindex;
devices[curindex].type = 0xC0DEE11C;
devices[curindex].availableModes[0].type = CELL_AUDIO_IN_CODING_TYPE_LPCM;
devices[curindex].availableModes[0].channel = CELL_AUDIO_IN_CHNUM_2;
devices[curindex].availableModes[0].fs = CELL_AUDIO_IN_FS_8KHZ | CELL_AUDIO_IN_FS_12KHZ | CELL_AUDIO_IN_FS_16KHZ | CELL_AUDIO_IN_FS_24KHZ | CELL_AUDIO_IN_FS_32KHZ | CELL_AUDIO_IN_FS_48KHZ;
devices[curindex].deviceNumber = curindex;
strcpy(devices[curindex].name, mic_list[index].c_str());
curindex++;
}
break;
case microphone_handler::real_singstar:
case microphone_handler::singstar:
// Only one device for singstar device
devices.emplace_back();
devices[curindex].portType = CELL_AUDIO_IN_PORT_USB;
devices[curindex].availableModeCount = 1;
devices[curindex].state = CELL_AUDIO_IN_DEVICE_STATE_AVAILABLE;
devices[curindex].deviceId = 0x57A3C0DE;
devices[curindex].type = 0xC0DE57A3;
devices[curindex].availableModes[0].type = CELL_AUDIO_IN_CODING_TYPE_LPCM;
devices[curindex].availableModes[0].channel = CELL_AUDIO_IN_CHNUM_2;
devices[curindex].availableModes[0].fs = CELL_AUDIO_IN_FS_8KHZ | CELL_AUDIO_IN_FS_12KHZ | CELL_AUDIO_IN_FS_16KHZ | CELL_AUDIO_IN_FS_24KHZ | CELL_AUDIO_IN_FS_32KHZ | CELL_AUDIO_IN_FS_48KHZ;
devices[curindex].deviceNumber = curindex;
strcpy(devices[curindex].name, mic_list[0].c_str());
curindex++;
break;
case microphone_handler::rocksmith:
devices.emplace_back();
devices[curindex].portType = CELL_AUDIO_IN_PORT_USB;
devices[curindex].availableModeCount = 1;
devices[curindex].state = CELL_AUDIO_IN_DEVICE_STATE_AVAILABLE;
devices[curindex].deviceId = 0x12BA00FF; // Specific to rocksmith usb input
devices[curindex].type = 0xC0DE73C4;
devices[curindex].availableModes[0].type = CELL_AUDIO_IN_CODING_TYPE_LPCM;
devices[curindex].availableModes[0].channel = CELL_AUDIO_IN_CHNUM_1;
devices[curindex].availableModes[0].fs = CELL_AUDIO_IN_FS_8KHZ | CELL_AUDIO_IN_FS_12KHZ | CELL_AUDIO_IN_FS_16KHZ | CELL_AUDIO_IN_FS_24KHZ | CELL_AUDIO_IN_FS_32KHZ | CELL_AUDIO_IN_FS_48KHZ;
devices[curindex].deviceNumber = curindex;
strcpy(devices[curindex].name, mic_list[0].c_str());
curindex++;
break;
default: break;
}
}
if (g_cfg.io.camera != camera_handler::null)
{
devices.emplace_back();
devices[curindex].portType = CELL_AUDIO_IN_PORT_USB;
devices[curindex].availableModeCount = 1;
devices[curindex].state = CELL_AUDIO_IN_DEVICE_STATE_AVAILABLE;
devices[curindex].deviceId = 0xDEADBEEF;
devices[curindex].type = 0xBEEFDEAD;
devices[curindex].availableModes[0].type = CELL_AUDIO_IN_CODING_TYPE_LPCM;
devices[curindex].availableModes[0].channel = CELL_AUDIO_IN_CHNUM_NONE;
devices[curindex].availableModes[0].fs = CELL_AUDIO_IN_FS_8KHZ | CELL_AUDIO_IN_FS_12KHZ | CELL_AUDIO_IN_FS_16KHZ | CELL_AUDIO_IN_FS_24KHZ | CELL_AUDIO_IN_FS_32KHZ | CELL_AUDIO_IN_FS_48KHZ;
devices[curindex].deviceNumber = curindex;
strcpy(devices[curindex].name, "USB Camera");
curindex++;
}
}
void avconf_manager::copy_device_info(u32 num, vm::ptr<CellAudioInDeviceInfo> info)
{
memset(info.get_ptr(), 0, sizeof(CellAudioInDeviceInfo));
info->portType = devices[num].portType;
info->availableModeCount = devices[num].availableModeCount;
info->state = devices[num].state;
info->deviceId = devices[num].deviceId;
info->type = devices[num].type;
info->availableModes[0].type = devices[num].availableModes[0].type;
info->availableModes[0].channel = devices[num].availableModes[0].channel;
info->availableModes[0].fs = devices[num].availableModes[0].fs;
info->deviceNumber = devices[num].deviceNumber;
strcpy(info->name, devices[num].name);
}
2019-11-15 20:15:00 +01:00
error_code cellAudioOutUnregisterDevice(u32 deviceNumber)
{
2017-08-09 18:40:11 +02:00
cellAvconfExt.todo("cellAudioOutUnregisterDevice(deviceNumber=0x%x)", deviceNumber);
return CELL_OK;
}
2019-11-15 20:15:00 +01:00
error_code cellAudioOutGetDeviceInfo2(u32 deviceNumber, u32 deviceIndex, vm::ptr<CellAudioOutDeviceInfo2> info)
{
2017-08-09 18:40:11 +02:00
cellAvconfExt.todo("cellAudioOutGetDeviceInfo2(deviceNumber=0x%x, deviceIndex=0x%x, info=*0x%x)", deviceNumber, deviceIndex, info);
return CELL_OK;
}
2019-11-15 20:15:00 +01:00
error_code cellVideoOutSetXVColor()
{
2017-08-09 18:40:11 +02:00
UNIMPLEMENTED_FUNC(cellAvconfExt);
return CELL_OK;
}
2019-11-15 20:15:00 +01:00
error_code cellVideoOutSetupDisplay()
{
2017-08-09 18:40:11 +02:00
UNIMPLEMENTED_FUNC(cellAvconfExt);
return CELL_OK;
}
2019-11-15 20:15:00 +01:00
error_code cellAudioInGetDeviceInfo(u32 deviceNumber, u32 deviceIndex, vm::ptr<CellAudioInDeviceInfo> info)
{
2017-08-09 18:40:11 +02:00
cellAvconfExt.todo("cellAudioInGetDeviceInfo(deviceNumber=0x%x, deviceIndex=0x%x, info=*0x%x)", deviceNumber, deviceIndex, info);
2018-12-20 23:36:56 +01:00
2019-08-22 02:52:47 +02:00
auto av_manager = g_fxo->get<avconf_manager>();
2018-12-20 23:36:56 +01:00
if (deviceNumber >= av_manager->devices.size())
return CELL_AUDIO_OUT_ERROR_DEVICE_NOT_FOUND;
av_manager->copy_device_info(deviceNumber, info);
return CELL_OK;
}
2019-11-15 20:15:00 +01:00
error_code cellVideoOutConvertCursorColor(u32 videoOut, s32 displaybuffer_format, f32 gamma, s32 source_buffer_format, vm::ptr<void> src_addr, vm::ptr<u32> dest_addr, s32 num)
{
2018-12-20 23:36:56 +01:00
cellAvconfExt.todo("cellVideoOutConvertCursorColor(videoOut=%d, displaybuffer_format=0x%x, gamma=0x%x, source_buffer_format=0x%x, src_addr=*0x%x, dest_addr=*0x%x, num=0x%x)", videoOut,
displaybuffer_format, gamma, source_buffer_format, src_addr, dest_addr, num);
return CELL_OK;
}
2019-11-15 20:15:00 +01:00
error_code cellVideoOutGetGamma(u32 videoOut, vm::ptr<f32> gamma)
{
cellAvconfExt.warning("cellVideoOutGetGamma(videoOut=%d, gamma=*0x%x)", videoOut, gamma);
if (videoOut != CELL_VIDEO_OUT_PRIMARY)
{
return CELL_VIDEO_OUT_ERROR_UNSUPPORTED_VIDEO_OUT;
}
2019-08-25 14:04:20 +02:00
auto conf = g_fxo->get<rsx::avconf>();
2018-12-20 23:36:56 +01:00
*gamma = conf->gamma;
return CELL_OK;
}
2019-11-15 20:15:00 +01:00
error_code cellAudioInGetAvailableDeviceInfo(u32 count, vm::ptr<CellAudioInDeviceInfo> device_info)
{
2018-12-20 23:36:56 +01:00
cellAvconfExt.todo("cellAudioInGetAvailableDeviceInfo(count=0x%x, info=*0x%x)", count, device_info);
2020-01-22 22:36:14 +01:00
if (count > 16 || !device_info)
2018-12-20 23:36:56 +01:00
{
return CELL_AUDIO_IN_ERROR_ILLEGAL_PARAMETER;
}
2019-08-22 02:52:47 +02:00
auto av_manager = g_fxo->get<avconf_manager>();
2018-12-20 23:36:56 +01:00
2019-12-01 18:14:58 +01:00
u32 num_devices_returned = std::min<u32>(count, ::size32(av_manager->devices));
2018-12-20 23:36:56 +01:00
for (u32 index = 0; index < num_devices_returned; index++)
{
av_manager->copy_device_info(index, device_info + index);
}
2019-12-01 18:14:58 +01:00
return not_an_error(num_devices_returned);
}
2019-11-15 20:15:00 +01:00
error_code cellAudioOutGetAvailableDeviceInfo(u32 count, vm::ptr<CellAudioOutDeviceInfo2> info)
{
2017-08-09 18:40:11 +02:00
cellAvconfExt.todo("cellAudioOutGetAvailableDeviceInfo(count=0x%x, info=*0x%x)", count, info);
2019-11-15 20:15:00 +01:00
return not_an_error(0); // number of available devices
}
2019-11-15 20:15:00 +01:00
error_code cellVideoOutSetGamma(u32 videoOut, f32 gamma)
{
cellAvconfExt.warning("cellVideoOutSetGamma(videoOut=%d, gamma=%f)", videoOut, gamma);
2019-11-15 20:56:26 +01:00
if (!(gamma >= 0.8f && gamma <= 1.2f))
{
2019-11-15 20:56:26 +01:00
return CELL_VIDEO_OUT_ERROR_ILLEGAL_PARAMETER;
}
2019-11-15 20:56:26 +01:00
if (videoOut != CELL_VIDEO_OUT_PRIMARY)
{
2019-11-15 20:56:26 +01:00
return CELL_VIDEO_OUT_ERROR_UNSUPPORTED_VIDEO_OUT;
}
2019-08-25 14:04:20 +02:00
auto conf = g_fxo->get<rsx::avconf>();
conf->gamma = gamma;
return CELL_OK;
}
2019-11-15 20:15:00 +01:00
error_code cellAudioOutRegisterDevice(u64 deviceType, vm::cptr<char> name, vm::ptr<CellAudioOutRegistrationOption> option, vm::ptr<CellAudioOutDeviceConfiguration> config)
{
2017-08-09 18:40:11 +02:00
cellAvconfExt.todo("cellAudioOutRegisterDevice(deviceType=0x%llx, name=%s, option=*0x%x, config=*0x%x)", deviceType, name, option, config);
2019-11-15 20:15:00 +01:00
return not_an_error(0); // device number
}
2019-11-15 20:15:00 +01:00
error_code cellAudioOutSetDeviceMode(u32 deviceMode)
{
2017-08-09 18:40:11 +02:00
cellAvconfExt.todo("cellAudioOutSetDeviceMode(deviceMode=0x%x)", deviceMode);
return CELL_OK;
}
2019-11-15 20:15:00 +01:00
error_code cellAudioInSetDeviceMode(u32 deviceMode)
{
2017-08-09 18:40:11 +02:00
cellAvconfExt.todo("cellAudioInSetDeviceMode(deviceMode=0x%x)", deviceMode);
return CELL_OK;
}
2019-11-15 20:15:00 +01:00
error_code cellAudioInRegisterDevice(u64 deviceType, vm::cptr<char> name, vm::ptr<CellAudioInRegistrationOption> option, vm::ptr<CellAudioInDeviceConfiguration> config)
{
2017-08-09 18:40:11 +02:00
cellAvconfExt.todo("cellAudioInRegisterDevice(deviceType=0x%llx, name=%s, option=*0x%x, config=*0x%x)", deviceType, name, option, config);
2018-12-20 23:36:56 +01:00
// option must be null, volume can be 1 (soft) to 5 (loud) (raises question about check for volume = 0)
if (option || !config || !name || config->volume > 5)
2019-11-15 20:15:00 +01:00
{
return CELL_AUDIO_IN_ERROR_ILLEGAL_PARAMETER;
}
return not_an_error(0); // device number
}
2019-11-15 20:15:00 +01:00
error_code cellAudioInUnregisterDevice(u32 deviceNumber)
{
2017-08-09 18:40:11 +02:00
cellAvconfExt.todo("cellAudioInUnregisterDevice(deviceNumber=0x%x)", deviceNumber);
return CELL_OK;
}
2019-11-15 20:15:00 +01:00
error_code cellVideoOutGetScreenSize(u32 videoOut, vm::ptr<f32> screenSize)
{
cellAvconfExt.warning("cellVideoOutGetScreenSize(videoOut=%d, screenSize=*0x%x)", videoOut, screenSize);
2015-01-02 16:29:57 +01:00
if (videoOut != CELL_VIDEO_OUT_PRIMARY)
{
return CELL_VIDEO_OUT_ERROR_UNSUPPORTED_VIDEO_OUT;
2015-01-02 16:29:57 +01:00
}
2018-12-20 23:36:56 +01:00
// TODO: Use virtual screen size
#ifdef _WIN32
2018-12-20 23:36:56 +01:00
// HDC screen = GetDC(NULL);
// float diagonal = roundf(sqrtf((powf(float(GetDeviceCaps(screen, HORZSIZE)), 2) + powf(float(GetDeviceCaps(screen, VERTSIZE)), 2))) * 0.0393f);
#else
// TODO: Linux implementation, without using wx
2015-01-02 16:29:57 +01:00
// float diagonal = roundf(sqrtf((powf(wxGetDisplaySizeMM().GetWidth(), 2) + powf(wxGetDisplaySizeMM().GetHeight(), 2))) * 0.0393f);
#endif
return CELL_VIDEO_OUT_ERROR_VALUE_IS_NOT_SET;
}
2019-11-15 20:15:00 +01:00
error_code cellVideoOutSetCopyControl(u32 videoOut, u32 control)
{
cellAvconfExt.todo("cellVideoOutSetCopyControl(videoOut=%d, control=0x%x)", videoOut, control);
return CELL_OK;
}
2018-12-20 23:36:56 +01:00
DECLARE(ppu_module_manager::cellAvconfExt)
("cellSysutilAvconfExt", []() {
2016-03-21 20:43:03 +01:00
REG_FUNC(cellSysutilAvconfExt, cellAudioOutUnregisterDevice);
REG_FUNC(cellSysutilAvconfExt, cellAudioOutGetDeviceInfo2);
REG_FUNC(cellSysutilAvconfExt, cellVideoOutSetXVColor);
REG_FUNC(cellSysutilAvconfExt, cellVideoOutSetupDisplay);
REG_FUNC(cellSysutilAvconfExt, cellAudioInGetDeviceInfo);
REG_FUNC(cellSysutilAvconfExt, cellVideoOutConvertCursorColor);
REG_FUNC(cellSysutilAvconfExt, cellVideoOutGetGamma);
REG_FUNC(cellSysutilAvconfExt, cellAudioInGetAvailableDeviceInfo);
REG_FUNC(cellSysutilAvconfExt, cellAudioOutGetAvailableDeviceInfo);
REG_FUNC(cellSysutilAvconfExt, cellVideoOutSetGamma);
REG_FUNC(cellSysutilAvconfExt, cellAudioOutRegisterDevice);
REG_FUNC(cellSysutilAvconfExt, cellAudioOutSetDeviceMode);
REG_FUNC(cellSysutilAvconfExt, cellAudioInSetDeviceMode);
REG_FUNC(cellSysutilAvconfExt, cellAudioInRegisterDevice);
REG_FUNC(cellSysutilAvconfExt, cellAudioInUnregisterDevice);
REG_FUNC(cellSysutilAvconfExt, cellVideoOutGetScreenSize);
REG_FUNC(cellSysutilAvconfExt, cellVideoOutSetCopyControl);
});