mirror of
https://github.com/wheremyfoodat/Panda3DS.git
synced 2025-04-06 14:15:41 +12:00
Merge branch 'master' into preempt-threads
This commit is contained in:
commit
b17284b23a
13 changed files with 191 additions and 26 deletions
33
.github/workflows/Hydra_Build.yml
vendored
33
.github/workflows/Hydra_Build.yml
vendored
|
@ -180,3 +180,36 @@ jobs:
|
||||||
with:
|
with:
|
||||||
name: Android Hydra core
|
name: Android Hydra core
|
||||||
path: '${{github.workspace}}/build/libAlber.so'
|
path: '${{github.workspace}}/build/libAlber.so'
|
||||||
|
|
||||||
|
ARM-Libretro:
|
||||||
|
runs-on: ubuntu-24.04-arm
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
- name: Fetch submodules
|
||||||
|
run: git submodule update --init --recursive
|
||||||
|
|
||||||
|
- name: Install misc packages
|
||||||
|
run: |
|
||||||
|
sudo apt-get update && sudo apt install libx11-dev libxext-dev libgl1 libglx-mesa0 mesa-common-dev libfuse2 libwayland-dev
|
||||||
|
|
||||||
|
- name: Install newer Clang
|
||||||
|
run: |
|
||||||
|
wget https://apt.llvm.org/llvm.sh
|
||||||
|
chmod +x ./llvm.sh
|
||||||
|
sudo ./llvm.sh 17
|
||||||
|
|
||||||
|
- name: Configure CMake
|
||||||
|
run: |
|
||||||
|
cmake -B ${{github.workspace}}/build -DCMAKE_BUILD_TYPE=${{env.BUILD_TYPE}} -DCMAKE_C_COMPILER=clang-17 -DCMAKE_CXX_COMPILER=clang++-17 -DENABLE_USER_BUILD=ON -DBUILD_LIBRETRO_CORE=ON -DENABLE_VULKAN=OFF -DCRYPTOPP_OPT_DISABLE_ASM=ON
|
||||||
|
|
||||||
|
- name: Build
|
||||||
|
run: cmake --build ${{github.workspace}}/build --config ${{env.BUILD_TYPE}}
|
||||||
|
|
||||||
|
- name: Upload Libretro core
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: Linux arm64 Libretro core
|
||||||
|
path: |
|
||||||
|
${{github.workspace}}/build/panda3ds_libretro.so
|
||||||
|
${{github.workspace}}/docs/libretro/panda3ds_libretro.info
|
||||||
|
|
|
@ -64,6 +64,7 @@ option(BUILD_HYDRA_CORE "Build a Hydra core" OFF)
|
||||||
option(BUILD_LIBRETRO_CORE "Build a Libretro core" OFF)
|
option(BUILD_LIBRETRO_CORE "Build a Libretro core" OFF)
|
||||||
option(ENABLE_RENDERDOC_API "Build with support for Renderdoc's capture API for graphics debugging" ON)
|
option(ENABLE_RENDERDOC_API "Build with support for Renderdoc's capture API for graphics debugging" ON)
|
||||||
option(DISABLE_SSE4 "Build with SSE4 instructions disabled, may reduce performance" OFF)
|
option(DISABLE_SSE4 "Build with SSE4 instructions disabled, may reduce performance" OFF)
|
||||||
|
option(USE_LIBRETRO_AUDIO "Enable to use the LR audio device with the LR core. Otherwise our own device is used" OFF)
|
||||||
|
|
||||||
set(OPENGL_PROFILE ${DEFAULT_OPENGL_PROFILE} CACHE STRING "OpenGL profile to use if OpenGL is enabled. Valid values are 'OpenGL' and 'OpenGLES'.")
|
set(OPENGL_PROFILE ${DEFAULT_OPENGL_PROFILE} CACHE STRING "OpenGL profile to use if OpenGL is enabled. Valid values are 'OpenGL' and 'OpenGLES'.")
|
||||||
set_property(CACHE OPENGL_PROFILE PROPERTY STRINGS OpenGL OpenGLES)
|
set_property(CACHE OPENGL_PROFILE PROPERTY STRINGS OpenGL OpenGLES)
|
||||||
|
@ -80,6 +81,10 @@ endif()
|
||||||
if(BUILD_LIBRETRO_CORE)
|
if(BUILD_LIBRETRO_CORE)
|
||||||
set(CMAKE_POSITION_INDEPENDENT_CODE ON)
|
set(CMAKE_POSITION_INDEPENDENT_CODE ON)
|
||||||
add_compile_definitions(__LIBRETRO__)
|
add_compile_definitions(__LIBRETRO__)
|
||||||
|
|
||||||
|
if(USE_LIBRETRO_AUDIO)
|
||||||
|
add_compile_definitions(USE_LIBRETRO_AUDIO_DEVICE)
|
||||||
|
endif()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(CMAKE_CXX_COMPILER_ID STREQUAL "MSVC" AND ENABLE_USER_BUILD)
|
if(CMAKE_CXX_COMPILER_ID STREQUAL "MSVC" AND ENABLE_USER_BUILD)
|
||||||
|
@ -157,7 +162,6 @@ if(ENABLE_DISCORD_RPC AND NOT ANDROID)
|
||||||
include_directories(third_party/discord-rpc/include)
|
include_directories(third_party/discord-rpc/include)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
|
||||||
if (NOT ANDROID)
|
if (NOT ANDROID)
|
||||||
if (USE_SYSTEM_SDL2)
|
if (USE_SYSTEM_SDL2)
|
||||||
find_package(SDL2 CONFIG REQUIRED)
|
find_package(SDL2 CONFIG REQUIRED)
|
||||||
|
@ -402,7 +406,8 @@ set(HEADER_FILES include/emulator.hpp include/helpers.hpp include/termcolor.hpp
|
||||||
include/align.hpp include/audio/aac_decoder.hpp include/PICA/pica_simd.hpp include/services/fonts.hpp
|
include/align.hpp include/audio/aac_decoder.hpp include/PICA/pica_simd.hpp include/services/fonts.hpp
|
||||||
include/audio/audio_interpolation.hpp include/audio/hle_mixer.hpp include/audio/dsp_simd.hpp
|
include/audio/audio_interpolation.hpp include/audio/hle_mixer.hpp include/audio/dsp_simd.hpp
|
||||||
include/services/dsp_firmware_db.hpp include/frontend_settings.hpp include/fs/archive_twl_photo.hpp
|
include/services/dsp_firmware_db.hpp include/frontend_settings.hpp include/fs/archive_twl_photo.hpp
|
||||||
include/fs/archive_twl_sound.hpp include/fs/archive_card_spi.hpp include/services/ns.hpp
|
include/fs/archive_twl_sound.hpp include/fs/archive_card_spi.hpp include/services/ns.hpp include/audio/audio_device.hpp
|
||||||
|
include/audio/audio_device_interface.hpp include/audio/libretro_audio_device.hpp
|
||||||
)
|
)
|
||||||
|
|
||||||
cmrc_add_resource_library(
|
cmrc_add_resource_library(
|
||||||
|
|
9
include/audio/audio_device.hpp
Normal file
9
include/audio/audio_device.hpp
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#if defined(__LIBRETRO__) && defined(USE_LIBRETRO_AUDIO_DEVICE)
|
||||||
|
#include "audio/libretro_audio_device.hpp"
|
||||||
|
using AudioDevice = LibretroAudioDevice;
|
||||||
|
#else
|
||||||
|
#include "audio/miniaudio_device.hpp"
|
||||||
|
using AudioDevice = MiniAudioDevice;
|
||||||
|
#endif
|
36
include/audio/audio_device_interface.hpp
Normal file
36
include/audio/audio_device_interface.hpp
Normal file
|
@ -0,0 +1,36 @@
|
||||||
|
#pragma once
|
||||||
|
#include <array>
|
||||||
|
|
||||||
|
#include "config.hpp"
|
||||||
|
#include "helpers.hpp"
|
||||||
|
#include "ring_buffer.hpp"
|
||||||
|
|
||||||
|
class AudioDeviceInterface {
|
||||||
|
protected:
|
||||||
|
static constexpr usize maxFrameCount = 0x2000;
|
||||||
|
|
||||||
|
using Samples = Common::RingBuffer<s16, maxFrameCount * 2>;
|
||||||
|
using RenderBatchCallback = usize (*)(const s16*, usize);
|
||||||
|
|
||||||
|
Samples* samples = nullptr;
|
||||||
|
|
||||||
|
const AudioDeviceConfig& audioSettings;
|
||||||
|
// Store the last stereo sample we output. We play this when underruning to avoid pops.
|
||||||
|
std::array<s16, 2> lastStereoSample{};
|
||||||
|
|
||||||
|
public:
|
||||||
|
AudioDeviceInterface(Samples* samples, const AudioDeviceConfig& audioSettings) : samples(samples), audioSettings(audioSettings) {}
|
||||||
|
|
||||||
|
bool running = false;
|
||||||
|
Samples* getSamples() { return samples; }
|
||||||
|
|
||||||
|
// If safe is on, we create a null audio device
|
||||||
|
virtual void init(Samples& samples, bool safe = false) = 0;
|
||||||
|
virtual void close() = 0;
|
||||||
|
|
||||||
|
virtual void start() = 0;
|
||||||
|
virtual void stop() = 0;
|
||||||
|
|
||||||
|
// Only used for audio devices that render multiple audio frames in one go, eg the libretro audio device.
|
||||||
|
virtual void renderBatch(RenderBatchCallback callback) {}
|
||||||
|
};
|
61
include/audio/libretro_audio_device.hpp
Normal file
61
include/audio/libretro_audio_device.hpp
Normal file
|
@ -0,0 +1,61 @@
|
||||||
|
#pragma once
|
||||||
|
#include <cstring>
|
||||||
|
|
||||||
|
#include "audio/audio_device_interface.hpp"
|
||||||
|
|
||||||
|
class LibretroAudioDevice final : public AudioDeviceInterface {
|
||||||
|
bool initialized = false;
|
||||||
|
|
||||||
|
public:
|
||||||
|
LibretroAudioDevice(const AudioDeviceConfig& audioSettings) : AudioDeviceInterface(nullptr, audioSettings), initialized(false) {
|
||||||
|
running = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void init(Samples& samples, bool safe = false) override {
|
||||||
|
this->samples = &samples;
|
||||||
|
|
||||||
|
initialized = true;
|
||||||
|
running = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void close() override {
|
||||||
|
initialized = false;
|
||||||
|
running = false;
|
||||||
|
};
|
||||||
|
|
||||||
|
void start() override { running = true; }
|
||||||
|
void stop() override { running = false; };
|
||||||
|
|
||||||
|
void renderBatch(RenderBatchCallback callback) override {
|
||||||
|
if (running) {
|
||||||
|
static constexpr usize sampleRate = 32768; // 3DS samples per second
|
||||||
|
static constexpr usize frameCount = sampleRate / 60; // 3DS samples per video frame
|
||||||
|
static constexpr usize channelCount = 2;
|
||||||
|
static s16 audioBuffer[frameCount * channelCount];
|
||||||
|
|
||||||
|
usize samplesWritten = 0;
|
||||||
|
samplesWritten += samples->pop(audioBuffer, frameCount * channelCount);
|
||||||
|
|
||||||
|
// Get the last sample for underrun handling
|
||||||
|
if (samplesWritten != 0) {
|
||||||
|
std::memcpy(&lastStereoSample[0], &audioBuffer[(samplesWritten - 1) * 2], sizeof(lastStereoSample));
|
||||||
|
}
|
||||||
|
|
||||||
|
// If underruning, copy the last output sample
|
||||||
|
{
|
||||||
|
s16* pointer = &audioBuffer[samplesWritten * 2];
|
||||||
|
s16 l = lastStereoSample[0];
|
||||||
|
s16 r = lastStereoSample[1];
|
||||||
|
|
||||||
|
for (usize i = samplesWritten; i < frameCount; i++) {
|
||||||
|
*pointer++ = l;
|
||||||
|
*pointer++ = r;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
callback(audioBuffer, sizeof(audioBuffer) / (channelCount * sizeof(s16)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool isInitialized() const { return initialized; }
|
||||||
|
};
|
|
@ -3,39 +3,31 @@
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
#include "config.hpp"
|
#include "audio/audio_device_interface.hpp"
|
||||||
#include "helpers.hpp"
|
|
||||||
#include "miniaudio.h"
|
#include "miniaudio.h"
|
||||||
#include "ring_buffer.hpp"
|
|
||||||
|
|
||||||
class MiniAudioDevice {
|
class MiniAudioDevice final : public AudioDeviceInterface {
|
||||||
using Samples = Common::RingBuffer<ma_int16, 0x2000 * 2>;
|
|
||||||
static constexpr ma_uint32 sampleRate = 32768; // 3DS sample rate
|
static constexpr ma_uint32 sampleRate = 32768; // 3DS sample rate
|
||||||
static constexpr ma_uint32 channelCount = 2; // Audio output is stereo
|
static constexpr ma_uint32 channelCount = 2; // Audio output is stereo
|
||||||
|
|
||||||
|
bool initialized = false;
|
||||||
|
|
||||||
ma_device device;
|
ma_device device;
|
||||||
ma_context context;
|
ma_context context;
|
||||||
ma_device_config deviceConfig;
|
ma_device_config deviceConfig;
|
||||||
Samples* samples = nullptr;
|
|
||||||
|
|
||||||
const AudioDeviceConfig& audioSettings;
|
|
||||||
|
|
||||||
bool initialized = false;
|
|
||||||
bool running = false;
|
|
||||||
|
|
||||||
// Store the last stereo sample we output. We play this when underruning to avoid pops.
|
// Store the last stereo sample we output. We play this when underruning to avoid pops.
|
||||||
std::array<s16, 2> lastStereoSample;
|
|
||||||
std::vector<std::string> audioDevices;
|
std::vector<std::string> audioDevices;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
MiniAudioDevice(const AudioDeviceConfig& audioSettings);
|
MiniAudioDevice(const AudioDeviceConfig& audioSettings);
|
||||||
|
|
||||||
// If safe is on, we create a null audio device
|
// If safe is on, we create a null audio device
|
||||||
void init(Samples& samples, bool safe = false);
|
void init(Samples& samples, bool safe = false) override;
|
||||||
void close();
|
void close() override;
|
||||||
|
|
||||||
void start();
|
void start() override;
|
||||||
void stop();
|
void stop() override;
|
||||||
|
|
||||||
bool isInitialized() const { return initialized; }
|
bool isInitialized() const { return initialized; }
|
||||||
};
|
};
|
|
@ -7,8 +7,8 @@
|
||||||
#include <span>
|
#include <span>
|
||||||
|
|
||||||
#include "PICA/gpu.hpp"
|
#include "PICA/gpu.hpp"
|
||||||
|
#include "audio/audio_device.hpp"
|
||||||
#include "audio/dsp_core.hpp"
|
#include "audio/dsp_core.hpp"
|
||||||
#include "audio/miniaudio_device.hpp"
|
|
||||||
#include "cheats.hpp"
|
#include "cheats.hpp"
|
||||||
#include "config.hpp"
|
#include "config.hpp"
|
||||||
#include "cpu.hpp"
|
#include "cpu.hpp"
|
||||||
|
@ -48,14 +48,14 @@ class Emulator {
|
||||||
Scheduler scheduler;
|
Scheduler scheduler;
|
||||||
|
|
||||||
Crypto::AESEngine aesEngine;
|
Crypto::AESEngine aesEngine;
|
||||||
MiniAudioDevice audioDevice;
|
AudioDevice audioDevice;
|
||||||
Cheats cheats;
|
Cheats cheats;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
static constexpr u32 width = 400;
|
static constexpr u32 width = 400;
|
||||||
static constexpr u32 height = 240 * 2; // * 2 because 2 screens
|
static constexpr u32 height = 240 * 2; // * 2 because 2 screens
|
||||||
ROMType romType = ROMType::None;
|
ROMType romType = ROMType::None;
|
||||||
bool running = false; // Is the emulator running a game?
|
bool running = false; // Is the emulator running a game?
|
||||||
|
|
||||||
private:
|
private:
|
||||||
#ifdef PANDA3DS_ENABLE_HTTP_SERVER
|
#ifdef PANDA3DS_ENABLE_HTTP_SERVER
|
||||||
|
@ -126,6 +126,7 @@ class Emulator {
|
||||||
LuaManager& getLua() { return lua; }
|
LuaManager& getLua() { return lua; }
|
||||||
Scheduler& getScheduler() { return scheduler; }
|
Scheduler& getScheduler() { return scheduler; }
|
||||||
Memory& getMemory() { return memory; }
|
Memory& getMemory() { return memory; }
|
||||||
|
AudioDeviceInterface& getAudioDevice() { return audioDevice; }
|
||||||
|
|
||||||
RendererType getRendererType() const { return config.rendererType; }
|
RendererType getRendererType() const { return config.rendererType; }
|
||||||
Renderer* getRenderer() { return gpu.getRenderer(); }
|
Renderer* getRenderer() { return gpu.getRenderer(); }
|
||||||
|
|
|
@ -23,6 +23,7 @@ class CFGService {
|
||||||
void getConfigInfoBlk2(u32 messagePointer);
|
void getConfigInfoBlk2(u32 messagePointer);
|
||||||
void getConfigInfoBlk8(u32 messagePointer, u32 commandWord);
|
void getConfigInfoBlk8(u32 messagePointer, u32 commandWord);
|
||||||
void getCountryCodeID(u32 messagePointer);
|
void getCountryCodeID(u32 messagePointer);
|
||||||
|
void getCountryCodeString(u32 messagePointer);
|
||||||
void getLocalFriendCodeSeed(u32 messagePointer);
|
void getLocalFriendCodeSeed(u32 messagePointer);
|
||||||
void getRegionCanadaUSA(u32 messagePointer);
|
void getRegionCanadaUSA(u32 messagePointer);
|
||||||
void getSystemModel(u32 messagePointer);
|
void getSystemModel(u32 messagePointer);
|
||||||
|
|
|
@ -117,7 +117,7 @@ Panda3DS also supports controller input using the SDL2 GameController API.
|
||||||
- [Kaizen](https://github.com/SimoneN64/Kaizen): Experimental work-in-progress low-level N64 emulator
|
- [Kaizen](https://github.com/SimoneN64/Kaizen): Experimental work-in-progress low-level N64 emulator
|
||||||
- [ChonkyStation](https://github.com/liuk7071/ChonkyStation): Work-in-progress PlayStation emulator
|
- [ChonkyStation](https://github.com/liuk7071/ChonkyStation): Work-in-progress PlayStation emulator
|
||||||
- [shadPS4](https://github.com/shadps4-emu/shadPS4): Work-in-progress PS4 emulator by the founder of PCSX, PCSX2 and more
|
- [shadPS4](https://github.com/shadps4-emu/shadPS4): Work-in-progress PS4 emulator by the founder of PCSX, PCSX2 and more
|
||||||
- [Hydra](https://github.com/hydra-emu/hydra): Cross-platform GameBoy, NES, N64 and Chip-8 emulator
|
- [felix86](https://github.com/OFFTKP/felix86): A new x86-64 → RISC-V Linux userspace emulator
|
||||||
- [Tanuki3DS](https://github.com/burhanr13/Tanuki3DS/): A new 3DS emulator for MacOS and Linux
|
- [Tanuki3DS](https://github.com/burhanr13/Tanuki3DS/): A new 3DS emulator for MacOS and Linux
|
||||||
# Support
|
# Support
|
||||||
If you find this project exciting and want to support the founder, check out [his Patreon](https://www.patreon.com/wheremyfoodat) or [Ko-fi](https://ko-fi.com/wheremyfoodat)
|
If you find this project exciting and want to support the founder, check out [his Patreon](https://www.patreon.com/wheremyfoodat) or [Ko-fi](https://ko-fi.com/wheremyfoodat)
|
||||||
|
|
|
@ -7,8 +7,9 @@
|
||||||
|
|
||||||
#include "helpers.hpp"
|
#include "helpers.hpp"
|
||||||
|
|
||||||
MiniAudioDevice::MiniAudioDevice(const AudioDeviceConfig& audioSettings)
|
MiniAudioDevice::MiniAudioDevice(const AudioDeviceConfig& audioSettings) : AudioDeviceInterface(nullptr, audioSettings), initialized(false) {
|
||||||
: initialized(false), running(false), samples(nullptr), audioSettings(audioSettings) {}
|
running = false;
|
||||||
|
}
|
||||||
|
|
||||||
void MiniAudioDevice::init(Samples& samples, bool safe) {
|
void MiniAudioDevice::init(Samples& samples, bool safe) {
|
||||||
this->samples = &samples;
|
this->samples = &samples;
|
||||||
|
@ -212,4 +213,4 @@ void MiniAudioDevice::close() {
|
||||||
ma_device_uninit(&device);
|
ma_device_uninit(&device);
|
||||||
ma_context_uninit(&context);
|
ma_context_uninit(&context);
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -18,6 +18,7 @@ namespace CFGCommands {
|
||||||
GetSystemModel = 0x00050000,
|
GetSystemModel = 0x00050000,
|
||||||
TranslateCountryInfo = 0x00080080,
|
TranslateCountryInfo = 0x00080080,
|
||||||
|
|
||||||
|
GetCountryCodeString = 0x00090040,
|
||||||
GetCountryCodeID = 0x000A0040,
|
GetCountryCodeID = 0x000A0040,
|
||||||
IsFangateSupported = 0x000B0000,
|
IsFangateSupported = 0x000B0000,
|
||||||
SetConfigInfoBlk4 = 0x04020082,
|
SetConfigInfoBlk4 = 0x04020082,
|
||||||
|
@ -50,6 +51,7 @@ void CFGService::handleSyncRequest(u32 messagePointer, CFGService::Type type) {
|
||||||
if (type != Type::NOR) {
|
if (type != Type::NOR) {
|
||||||
switch (command) {
|
switch (command) {
|
||||||
case CFGCommands::GetConfigInfoBlk2: [[likely]] getConfigInfoBlk2(messagePointer); break;
|
case CFGCommands::GetConfigInfoBlk2: [[likely]] getConfigInfoBlk2(messagePointer); break;
|
||||||
|
case CFGCommands::GetCountryCodeString: getCountryCodeString(messagePointer); break;
|
||||||
case CFGCommands::GetCountryCodeID: getCountryCodeID(messagePointer); break;
|
case CFGCommands::GetCountryCodeID: getCountryCodeID(messagePointer); break;
|
||||||
case CFGCommands::GetRegionCanadaUSA: getRegionCanadaUSA(messagePointer); break;
|
case CFGCommands::GetRegionCanadaUSA: getRegionCanadaUSA(messagePointer); break;
|
||||||
case CFGCommands::GetSystemModel: getSystemModel(messagePointer); break;
|
case CFGCommands::GetSystemModel: getSystemModel(messagePointer); break;
|
||||||
|
@ -315,6 +317,24 @@ void CFGService::getCountryCodeID(u32 messagePointer) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void CFGService::getCountryCodeString(u32 messagePointer) {
|
||||||
|
const u16 id = mem.read16(messagePointer + 4);
|
||||||
|
log("CFG::getCountryCodeString (id = %04X)\n", id);
|
||||||
|
|
||||||
|
mem.write32(messagePointer, IPC::responseHeader(0x09, 2, 0));
|
||||||
|
|
||||||
|
for (auto [string, code] : countryCodeToTableIDMap) {
|
||||||
|
if (code == id) {
|
||||||
|
mem.write32(messagePointer + 4, Result::Success);
|
||||||
|
mem.write32(messagePointer + 8, u32(string));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Code is not a valid country code, return an appropriate error
|
||||||
|
mem.write32(messagePointer + 4, 0xD90103FA);
|
||||||
|
}
|
||||||
|
|
||||||
void CFGService::secureInfoGetByte101(u32 messagePointer) {
|
void CFGService::secureInfoGetByte101(u32 messagePointer) {
|
||||||
log("CFG::SecureInfoGetByte101\n");
|
log("CFG::SecureInfoGetByte101\n");
|
||||||
|
|
||||||
|
|
|
@ -19,7 +19,11 @@ void HTTPService::handleSyncRequest(u32 messagePointer) {
|
||||||
case HTTPCommands::CreateRootCertChain: createRootCertChain(messagePointer); break;
|
case HTTPCommands::CreateRootCertChain: createRootCertChain(messagePointer); break;
|
||||||
case HTTPCommands::Initialize: initialize(messagePointer); break;
|
case HTTPCommands::Initialize: initialize(messagePointer); break;
|
||||||
case HTTPCommands::RootCertChainAddDefaultCert: rootCertChainAddDefaultCert(messagePointer); break;
|
case HTTPCommands::RootCertChainAddDefaultCert: rootCertChainAddDefaultCert(messagePointer); break;
|
||||||
default: Helpers::panic("HTTP service requested. Command: %08X\n", command);
|
|
||||||
|
default:
|
||||||
|
Helpers::warn("HTTP service requested. Command: %08X\n", command);
|
||||||
|
mem.write32(messagePointer + 4, Result::FailurePlaceholder);
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -389,6 +389,8 @@ void retro_run() {
|
||||||
emulator->runFrame();
|
emulator->runFrame();
|
||||||
|
|
||||||
videoCallback(RETRO_HW_FRAME_BUFFER_VALID, emulator->width, emulator->height, 0);
|
videoCallback(RETRO_HW_FRAME_BUFFER_VALID, emulator->width, emulator->height, 0);
|
||||||
|
// Call audio batch callback
|
||||||
|
emulator->getAudioDevice().renderBatch(audioBatchCallback);
|
||||||
}
|
}
|
||||||
|
|
||||||
void retro_set_controller_port_device(uint port, uint device) {}
|
void retro_set_controller_port_device(uint port, uint device) {}
|
||||||
|
|
Loading…
Add table
Reference in a new issue