2020-05-08 22:32:29 +02:00
|
|
|
/*
|
|
|
|
DingusPPC - The Experimental PowerPC Macintosh emulator
|
2023-04-23 21:38:48 +02:00
|
|
|
Copyright (C) 2018-23 divingkatae and maximum
|
2020-05-08 22:32:29 +02:00
|
|
|
(theweirdo) spatium
|
|
|
|
|
|
|
|
(Contact divingkatae#1017 or powermax#2286 on Discord for more info)
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2024-11-09 22:25:48 -08:00
|
|
|
#ifndef NOMINMAX
|
|
|
|
#define NOMINMAX
|
|
|
|
#endif // NOMINMAX
|
|
|
|
|
|
|
|
#include <core/timermanager.h>
|
|
|
|
#include <cpu/ppc/ppcemu.h>
|
2021-10-23 20:17:47 +02:00
|
|
|
#include <devices/common/dmacore.h>
|
|
|
|
#include <devices/sound/soundserver.h>
|
|
|
|
#include <endianswap.h>
|
|
|
|
|
2024-11-09 22:25:48 -08:00
|
|
|
#include <algorithm>
|
|
|
|
#include <functional>
|
2021-09-16 00:46:38 +02:00
|
|
|
#include <loguru.hpp>
|
2020-05-15 02:36:40 +02:00
|
|
|
#include <cubeb/cubeb.h>
|
2020-05-18 02:51:56 +02:00
|
|
|
#ifdef _WIN32
|
|
|
|
#include <objbase.h>
|
|
|
|
#endif
|
2020-05-08 22:32:29 +02:00
|
|
|
|
2024-11-09 22:25:48 -08:00
|
|
|
typedef enum {
|
Make Emscripten build not depend on SDL2 or cubeb
While Emscripten has an SDL compabtility layer, it assumes that the
code is executing in the main browser process (and thus has access to
them DOM). The Infinite Mac project runs emulators in a worker thread
(for better performance) and has a custom API for the display, sound,
input, etc. Similarly, it does not need the cross-platform sound support
from cubeb, there there is a sound API as well.
This commit makes SDL (*_sdl.cpp) and cubeb-based (*_cubeb.cpp) code be
skipped when targeting Emscripten, and instead *_js.cpp files are used
instead (this is the cross-platform convention used by Chromium[^1], and
could be extended for other targets).
For hostevents.cpp and soundserver.cpp the entire file was replaced,
whereas for videoctrl.cpp there was enough shared logic that it was
kept, and the platform-specific bits were moved behind a Display class
that can have per-platform implementations. For cases where we need
additional private fields in the platform-specific classes, we use
a PIMPL pattern.
The *_js.cpp files with implementations are not included in this
commit, since they are closely tied to the Infinite Mac project, and
will live in its fork of DingusPPC.
[^1]: https://www.chromium.org/developers/design-documents/conventions-and-patterns-for-multi-platform-development/
2023-09-03 17:59:35 -07:00
|
|
|
SND_SERVER_DOWN = 0,
|
|
|
|
SND_API_READY,
|
|
|
|
SND_SERVER_UP,
|
|
|
|
SND_STREAM_OPENED,
|
|
|
|
SND_STREAM_CLOSED
|
2024-11-09 22:25:48 -08:00
|
|
|
} Status;
|
Make Emscripten build not depend on SDL2 or cubeb
While Emscripten has an SDL compabtility layer, it assumes that the
code is executing in the main browser process (and thus has access to
them DOM). The Infinite Mac project runs emulators in a worker thread
(for better performance) and has a custom API for the display, sound,
input, etc. Similarly, it does not need the cross-platform sound support
from cubeb, there there is a sound API as well.
This commit makes SDL (*_sdl.cpp) and cubeb-based (*_cubeb.cpp) code be
skipped when targeting Emscripten, and instead *_js.cpp files are used
instead (this is the cross-platform convention used by Chromium[^1], and
could be extended for other targets).
For hostevents.cpp and soundserver.cpp the entire file was replaced,
whereas for videoctrl.cpp there was enough shared logic that it was
kept, and the platform-specific bits were moved behind a Display class
that can have per-platform implementations. For cases where we need
additional private fields in the platform-specific classes, we use
a PIMPL pattern.
The *_js.cpp files with implementations are not included in this
commit, since they are closely tied to the Infinite Mac project, and
will live in its fork of DingusPPC.
[^1]: https://www.chromium.org/developers/design-documents/conventions-and-patterns-for-multi-platform-development/
2023-09-03 17:59:35 -07:00
|
|
|
|
|
|
|
class SoundServer::Impl {
|
|
|
|
public:
|
2024-11-09 22:25:48 -08:00
|
|
|
Status status = SND_SERVER_DOWN;
|
Make Emscripten build not depend on SDL2 or cubeb
While Emscripten has an SDL compabtility layer, it assumes that the
code is executing in the main browser process (and thus has access to
them DOM). The Infinite Mac project runs emulators in a worker thread
(for better performance) and has a custom API for the display, sound,
input, etc. Similarly, it does not need the cross-platform sound support
from cubeb, there there is a sound API as well.
This commit makes SDL (*_sdl.cpp) and cubeb-based (*_cubeb.cpp) code be
skipped when targeting Emscripten, and instead *_js.cpp files are used
instead (this is the cross-platform convention used by Chromium[^1], and
could be extended for other targets).
For hostevents.cpp and soundserver.cpp the entire file was replaced,
whereas for videoctrl.cpp there was enough shared logic that it was
kept, and the platform-specific bits were moved behind a Display class
that can have per-platform implementations. For cases where we need
additional private fields in the platform-specific classes, we use
a PIMPL pattern.
The *_js.cpp files with implementations are not included in this
commit, since they are closely tied to the Infinite Mac project, and
will live in its fork of DingusPPC.
[^1]: https://www.chromium.org/developers/design-documents/conventions-and-patterns-for-multi-platform-development/
2023-09-03 17:59:35 -07:00
|
|
|
cubeb *cubeb_ctx;
|
|
|
|
cubeb_stream *out_stream;
|
2024-11-09 22:25:48 -08:00
|
|
|
|
|
|
|
uint32_t deterministic_poll_timer = 0;
|
|
|
|
std::function<void()> deterministic_poll_cb;
|
Make Emscripten build not depend on SDL2 or cubeb
While Emscripten has an SDL compabtility layer, it assumes that the
code is executing in the main browser process (and thus has access to
them DOM). The Infinite Mac project runs emulators in a worker thread
(for better performance) and has a custom API for the display, sound,
input, etc. Similarly, it does not need the cross-platform sound support
from cubeb, there there is a sound API as well.
This commit makes SDL (*_sdl.cpp) and cubeb-based (*_cubeb.cpp) code be
skipped when targeting Emscripten, and instead *_js.cpp files are used
instead (this is the cross-platform convention used by Chromium[^1], and
could be extended for other targets).
For hostevents.cpp and soundserver.cpp the entire file was replaced,
whereas for videoctrl.cpp there was enough shared logic that it was
kept, and the platform-specific bits were moved behind a Display class
that can have per-platform implementations. For cases where we need
additional private fields in the platform-specific classes, we use
a PIMPL pattern.
The *_js.cpp files with implementations are not included in this
commit, since they are closely tied to the Infinite Mac project, and
will live in its fork of DingusPPC.
[^1]: https://www.chromium.org/developers/design-documents/conventions-and-patterns-for-multi-platform-development/
2023-09-03 17:59:35 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
SoundServer::SoundServer(): impl(std::make_unique<Impl>())
|
|
|
|
{
|
|
|
|
supports_types(HWCompType::SND_SERVER);
|
|
|
|
this->start();
|
|
|
|
}
|
|
|
|
|
|
|
|
SoundServer::~SoundServer()
|
|
|
|
{
|
|
|
|
this->shutdown();
|
|
|
|
}
|
|
|
|
|
2020-05-15 02:36:40 +02:00
|
|
|
int SoundServer::start()
|
|
|
|
{
|
|
|
|
int res;
|
|
|
|
|
2020-05-18 02:51:56 +02:00
|
|
|
#ifdef _WIN32
|
|
|
|
CoInitialize(nullptr);
|
|
|
|
#endif
|
|
|
|
|
Make Emscripten build not depend on SDL2 or cubeb
While Emscripten has an SDL compabtility layer, it assumes that the
code is executing in the main browser process (and thus has access to
them DOM). The Infinite Mac project runs emulators in a worker thread
(for better performance) and has a custom API for the display, sound,
input, etc. Similarly, it does not need the cross-platform sound support
from cubeb, there there is a sound API as well.
This commit makes SDL (*_sdl.cpp) and cubeb-based (*_cubeb.cpp) code be
skipped when targeting Emscripten, and instead *_js.cpp files are used
instead (this is the cross-platform convention used by Chromium[^1], and
could be extended for other targets).
For hostevents.cpp and soundserver.cpp the entire file was replaced,
whereas for videoctrl.cpp there was enough shared logic that it was
kept, and the platform-specific bits were moved behind a Display class
that can have per-platform implementations. For cases where we need
additional private fields in the platform-specific classes, we use
a PIMPL pattern.
The *_js.cpp files with implementations are not included in this
commit, since they are closely tied to the Infinite Mac project, and
will live in its fork of DingusPPC.
[^1]: https://www.chromium.org/developers/design-documents/conventions-and-patterns-for-multi-platform-development/
2023-09-03 17:59:35 -07:00
|
|
|
impl->status = SND_SERVER_DOWN;
|
2020-05-15 02:36:40 +02:00
|
|
|
|
Make Emscripten build not depend on SDL2 or cubeb
While Emscripten has an SDL compabtility layer, it assumes that the
code is executing in the main browser process (and thus has access to
them DOM). The Infinite Mac project runs emulators in a worker thread
(for better performance) and has a custom API for the display, sound,
input, etc. Similarly, it does not need the cross-platform sound support
from cubeb, there there is a sound API as well.
This commit makes SDL (*_sdl.cpp) and cubeb-based (*_cubeb.cpp) code be
skipped when targeting Emscripten, and instead *_js.cpp files are used
instead (this is the cross-platform convention used by Chromium[^1], and
could be extended for other targets).
For hostevents.cpp and soundserver.cpp the entire file was replaced,
whereas for videoctrl.cpp there was enough shared logic that it was
kept, and the platform-specific bits were moved behind a Display class
that can have per-platform implementations. For cases where we need
additional private fields in the platform-specific classes, we use
a PIMPL pattern.
The *_js.cpp files with implementations are not included in this
commit, since they are closely tied to the Infinite Mac project, and
will live in its fork of DingusPPC.
[^1]: https://www.chromium.org/developers/design-documents/conventions-and-patterns-for-multi-platform-development/
2023-09-03 17:59:35 -07:00
|
|
|
res = cubeb_init(&impl->cubeb_ctx, "Dingus sound server", NULL);
|
2020-05-15 02:36:40 +02:00
|
|
|
if (res != CUBEB_OK) {
|
|
|
|
LOG_F(ERROR, "Could not initialize Cubeb library");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
Make Emscripten build not depend on SDL2 or cubeb
While Emscripten has an SDL compabtility layer, it assumes that the
code is executing in the main browser process (and thus has access to
them DOM). The Infinite Mac project runs emulators in a worker thread
(for better performance) and has a custom API for the display, sound,
input, etc. Similarly, it does not need the cross-platform sound support
from cubeb, there there is a sound API as well.
This commit makes SDL (*_sdl.cpp) and cubeb-based (*_cubeb.cpp) code be
skipped when targeting Emscripten, and instead *_js.cpp files are used
instead (this is the cross-platform convention used by Chromium[^1], and
could be extended for other targets).
For hostevents.cpp and soundserver.cpp the entire file was replaced,
whereas for videoctrl.cpp there was enough shared logic that it was
kept, and the platform-specific bits were moved behind a Display class
that can have per-platform implementations. For cases where we need
additional private fields in the platform-specific classes, we use
a PIMPL pattern.
The *_js.cpp files with implementations are not included in this
commit, since they are closely tied to the Infinite Mac project, and
will live in its fork of DingusPPC.
[^1]: https://www.chromium.org/developers/design-documents/conventions-and-patterns-for-multi-platform-development/
2023-09-03 17:59:35 -07:00
|
|
|
LOG_F(INFO, "Connected to backend: %s", cubeb_get_backend_id(impl->cubeb_ctx));
|
2020-05-15 02:36:40 +02:00
|
|
|
|
Make Emscripten build not depend on SDL2 or cubeb
While Emscripten has an SDL compabtility layer, it assumes that the
code is executing in the main browser process (and thus has access to
them DOM). The Infinite Mac project runs emulators in a worker thread
(for better performance) and has a custom API for the display, sound,
input, etc. Similarly, it does not need the cross-platform sound support
from cubeb, there there is a sound API as well.
This commit makes SDL (*_sdl.cpp) and cubeb-based (*_cubeb.cpp) code be
skipped when targeting Emscripten, and instead *_js.cpp files are used
instead (this is the cross-platform convention used by Chromium[^1], and
could be extended for other targets).
For hostevents.cpp and soundserver.cpp the entire file was replaced,
whereas for videoctrl.cpp there was enough shared logic that it was
kept, and the platform-specific bits were moved behind a Display class
that can have per-platform implementations. For cases where we need
additional private fields in the platform-specific classes, we use
a PIMPL pattern.
The *_js.cpp files with implementations are not included in this
commit, since they are closely tied to the Infinite Mac project, and
will live in its fork of DingusPPC.
[^1]: https://www.chromium.org/developers/design-documents/conventions-and-patterns-for-multi-platform-development/
2023-09-03 17:59:35 -07:00
|
|
|
impl->status = SND_API_READY;
|
2020-05-15 02:36:40 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SoundServer::shutdown()
|
|
|
|
{
|
Make Emscripten build not depend on SDL2 or cubeb
While Emscripten has an SDL compabtility layer, it assumes that the
code is executing in the main browser process (and thus has access to
them DOM). The Infinite Mac project runs emulators in a worker thread
(for better performance) and has a custom API for the display, sound,
input, etc. Similarly, it does not need the cross-platform sound support
from cubeb, there there is a sound API as well.
This commit makes SDL (*_sdl.cpp) and cubeb-based (*_cubeb.cpp) code be
skipped when targeting Emscripten, and instead *_js.cpp files are used
instead (this is the cross-platform convention used by Chromium[^1], and
could be extended for other targets).
For hostevents.cpp and soundserver.cpp the entire file was replaced,
whereas for videoctrl.cpp there was enough shared logic that it was
kept, and the platform-specific bits were moved behind a Display class
that can have per-platform implementations. For cases where we need
additional private fields in the platform-specific classes, we use
a PIMPL pattern.
The *_js.cpp files with implementations are not included in this
commit, since they are closely tied to the Infinite Mac project, and
will live in its fork of DingusPPC.
[^1]: https://www.chromium.org/developers/design-documents/conventions-and-patterns-for-multi-platform-development/
2023-09-03 17:59:35 -07:00
|
|
|
switch (impl->status) {
|
2022-07-17 05:50:32 +02:00
|
|
|
case SND_STREAM_OPENED:
|
|
|
|
close_out_stream();
|
|
|
|
/* fall through */
|
2023-10-18 16:18:54 +02:00
|
|
|
case SND_STREAM_CLOSED:
|
|
|
|
/* fall through */
|
2020-05-15 02:36:40 +02:00
|
|
|
case SND_SERVER_UP:
|
|
|
|
/* fall through */
|
|
|
|
case SND_API_READY:
|
Make Emscripten build not depend on SDL2 or cubeb
While Emscripten has an SDL compabtility layer, it assumes that the
code is executing in the main browser process (and thus has access to
them DOM). The Infinite Mac project runs emulators in a worker thread
(for better performance) and has a custom API for the display, sound,
input, etc. Similarly, it does not need the cross-platform sound support
from cubeb, there there is a sound API as well.
This commit makes SDL (*_sdl.cpp) and cubeb-based (*_cubeb.cpp) code be
skipped when targeting Emscripten, and instead *_js.cpp files are used
instead (this is the cross-platform convention used by Chromium[^1], and
could be extended for other targets).
For hostevents.cpp and soundserver.cpp the entire file was replaced,
whereas for videoctrl.cpp there was enough shared logic that it was
kept, and the platform-specific bits were moved behind a Display class
that can have per-platform implementations. For cases where we need
additional private fields in the platform-specific classes, we use
a PIMPL pattern.
The *_js.cpp files with implementations are not included in this
commit, since they are closely tied to the Infinite Mac project, and
will live in its fork of DingusPPC.
[^1]: https://www.chromium.org/developers/design-documents/conventions-and-patterns-for-multi-platform-development/
2023-09-03 17:59:35 -07:00
|
|
|
cubeb_destroy(impl->cubeb_ctx);
|
2024-11-09 22:25:48 -08:00
|
|
|
break;
|
|
|
|
case SND_SERVER_DOWN:
|
|
|
|
// Nothing to do.
|
|
|
|
break;
|
2020-05-15 02:36:40 +02:00
|
|
|
}
|
|
|
|
|
Make Emscripten build not depend on SDL2 or cubeb
While Emscripten has an SDL compabtility layer, it assumes that the
code is executing in the main browser process (and thus has access to
them DOM). The Infinite Mac project runs emulators in a worker thread
(for better performance) and has a custom API for the display, sound,
input, etc. Similarly, it does not need the cross-platform sound support
from cubeb, there there is a sound API as well.
This commit makes SDL (*_sdl.cpp) and cubeb-based (*_cubeb.cpp) code be
skipped when targeting Emscripten, and instead *_js.cpp files are used
instead (this is the cross-platform convention used by Chromium[^1], and
could be extended for other targets).
For hostevents.cpp and soundserver.cpp the entire file was replaced,
whereas for videoctrl.cpp there was enough shared logic that it was
kept, and the platform-specific bits were moved behind a Display class
that can have per-platform implementations. For cases where we need
additional private fields in the platform-specific classes, we use
a PIMPL pattern.
The *_js.cpp files with implementations are not included in this
commit, since they are closely tied to the Infinite Mac project, and
will live in its fork of DingusPPC.
[^1]: https://www.chromium.org/developers/design-documents/conventions-and-patterns-for-multi-platform-development/
2023-09-03 17:59:35 -07:00
|
|
|
impl->status = SND_SERVER_DOWN;
|
2020-05-15 02:36:40 +02:00
|
|
|
|
|
|
|
LOG_F(INFO, "Sound Server shut down.");
|
|
|
|
}
|
|
|
|
|
2021-10-04 23:46:19 +02:00
|
|
|
long sound_out_callback(cubeb_stream *stream, void *user_data,
|
|
|
|
void const *input_buffer, void *output_buffer,
|
|
|
|
long req_frames)
|
|
|
|
{
|
|
|
|
uint8_t *p_in;
|
|
|
|
int16_t* in_buf, * out_buf;
|
|
|
|
uint32_t got_len;
|
|
|
|
long frames, out_frames;
|
|
|
|
DmaOutChannel *dma_ch = static_cast<DmaOutChannel*>(user_data); /* C API baby! */
|
|
|
|
|
2023-11-21 20:36:43 -08:00
|
|
|
if (!dma_ch->is_out_active()) {
|
2021-10-04 23:46:19 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
out_buf = (int16_t*)output_buffer;
|
|
|
|
|
|
|
|
out_frames = 0;
|
|
|
|
|
|
|
|
while (req_frames > 0) {
|
2022-12-21 03:20:39 -08:00
|
|
|
if (!dma_ch->pull_data((uint32_t)req_frames << 2, &got_len, &p_in)) {
|
2024-05-04 22:57:39 -07:00
|
|
|
if ((in_buf = (int16_t*)p_in)) {
|
|
|
|
frames = got_len >> 2;
|
|
|
|
|
|
|
|
for (int i = (int)frames; i > 0; i--) {
|
|
|
|
out_buf[0] = BYTESWAP_16(in_buf[0]);
|
|
|
|
out_buf[1] = BYTESWAP_16(in_buf[1]);
|
|
|
|
in_buf += 2;
|
|
|
|
out_buf += 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
req_frames -= frames;
|
|
|
|
out_frames += frames;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
LOG_F(ERROR, "Didn't get qdata");
|
2021-10-04 23:46:19 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return out_frames;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void status_callback(cubeb_stream *stream, void *user_data, cubeb_state state)
|
|
|
|
{
|
|
|
|
LOG_F(9, "Cubeb status callback fired, status = %d", state);
|
|
|
|
}
|
2020-05-15 02:36:40 +02:00
|
|
|
|
2024-11-09 22:25:48 -08:00
|
|
|
int SoundServer::open_out_stream(uint32_t sample_rate, DmaOutChannel *dma_ch)
|
2020-05-15 02:36:40 +02:00
|
|
|
{
|
2024-11-09 22:25:48 -08:00
|
|
|
if (is_deterministic) {
|
|
|
|
impl->deterministic_poll_cb = [dma_ch] {
|
|
|
|
if (!dma_ch->is_out_active()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Drain the DMA buffer, but don't do anything else.
|
|
|
|
int req_size = std::max(dma_ch->get_pull_data_remaining(), 1024);
|
|
|
|
int out_size = 0;
|
|
|
|
while (req_size > 0) {
|
|
|
|
uint8_t *chunk;
|
|
|
|
uint32_t chunk_size;
|
|
|
|
if (!dma_ch->pull_data(req_size, &chunk_size, &chunk)) {
|
|
|
|
req_size -= chunk_size;
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
impl->status = SND_STREAM_OPENED;
|
|
|
|
LOG_F(9, "Deterministic sound output callback set up.");
|
|
|
|
return 0;
|
|
|
|
}
|
2020-05-15 02:36:40 +02:00
|
|
|
int res;
|
|
|
|
uint32_t latency_frames;
|
|
|
|
cubeb_stream_params params;
|
|
|
|
|
|
|
|
params.format = CUBEB_SAMPLE_S16NE;
|
|
|
|
params.rate = sample_rate;
|
|
|
|
params.channels = 2;
|
|
|
|
params.layout = CUBEB_LAYOUT_STEREO;
|
|
|
|
params.prefs = CUBEB_STREAM_PREF_NONE;
|
|
|
|
|
Make Emscripten build not depend on SDL2 or cubeb
While Emscripten has an SDL compabtility layer, it assumes that the
code is executing in the main browser process (and thus has access to
them DOM). The Infinite Mac project runs emulators in a worker thread
(for better performance) and has a custom API for the display, sound,
input, etc. Similarly, it does not need the cross-platform sound support
from cubeb, there there is a sound API as well.
This commit makes SDL (*_sdl.cpp) and cubeb-based (*_cubeb.cpp) code be
skipped when targeting Emscripten, and instead *_js.cpp files are used
instead (this is the cross-platform convention used by Chromium[^1], and
could be extended for other targets).
For hostevents.cpp and soundserver.cpp the entire file was replaced,
whereas for videoctrl.cpp there was enough shared logic that it was
kept, and the platform-specific bits were moved behind a Display class
that can have per-platform implementations. For cases where we need
additional private fields in the platform-specific classes, we use
a PIMPL pattern.
The *_js.cpp files with implementations are not included in this
commit, since they are closely tied to the Infinite Mac project, and
will live in its fork of DingusPPC.
[^1]: https://www.chromium.org/developers/design-documents/conventions-and-patterns-for-multi-platform-development/
2023-09-03 17:59:35 -07:00
|
|
|
res = cubeb_get_min_latency(impl->cubeb_ctx, ¶ms, &latency_frames);
|
2020-05-15 02:36:40 +02:00
|
|
|
if (res != CUBEB_OK) {
|
|
|
|
LOG_F(ERROR, "Could not get minimum latency, error: %d", res);
|
|
|
|
return -1;
|
|
|
|
} else {
|
2023-04-23 21:38:48 +02:00
|
|
|
LOG_F(9, "Minimum sound latency: %d frames", latency_frames);
|
2020-05-15 02:36:40 +02:00
|
|
|
}
|
|
|
|
|
Make Emscripten build not depend on SDL2 or cubeb
While Emscripten has an SDL compabtility layer, it assumes that the
code is executing in the main browser process (and thus has access to
them DOM). The Infinite Mac project runs emulators in a worker thread
(for better performance) and has a custom API for the display, sound,
input, etc. Similarly, it does not need the cross-platform sound support
from cubeb, there there is a sound API as well.
This commit makes SDL (*_sdl.cpp) and cubeb-based (*_cubeb.cpp) code be
skipped when targeting Emscripten, and instead *_js.cpp files are used
instead (this is the cross-platform convention used by Chromium[^1], and
could be extended for other targets).
For hostevents.cpp and soundserver.cpp the entire file was replaced,
whereas for videoctrl.cpp there was enough shared logic that it was
kept, and the platform-specific bits were moved behind a Display class
that can have per-platform implementations. For cases where we need
additional private fields in the platform-specific classes, we use
a PIMPL pattern.
The *_js.cpp files with implementations are not included in this
commit, since they are closely tied to the Infinite Mac project, and
will live in its fork of DingusPPC.
[^1]: https://www.chromium.org/developers/design-documents/conventions-and-patterns-for-multi-platform-development/
2023-09-03 17:59:35 -07:00
|
|
|
res = cubeb_stream_init(impl->cubeb_ctx, &impl->out_stream, "SndOut stream",
|
2020-05-15 02:36:40 +02:00
|
|
|
NULL, NULL, NULL, ¶ms, latency_frames,
|
2024-11-09 22:25:48 -08:00
|
|
|
sound_out_callback, status_callback, dma_ch);
|
2020-05-15 02:36:40 +02:00
|
|
|
if (res != CUBEB_OK) {
|
|
|
|
LOG_F(ERROR, "Could not open sound output stream, error: %d", res);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2023-04-23 21:38:48 +02:00
|
|
|
LOG_F(9, "Sound output stream opened.");
|
2020-05-15 02:36:40 +02:00
|
|
|
|
Make Emscripten build not depend on SDL2 or cubeb
While Emscripten has an SDL compabtility layer, it assumes that the
code is executing in the main browser process (and thus has access to
them DOM). The Infinite Mac project runs emulators in a worker thread
(for better performance) and has a custom API for the display, sound,
input, etc. Similarly, it does not need the cross-platform sound support
from cubeb, there there is a sound API as well.
This commit makes SDL (*_sdl.cpp) and cubeb-based (*_cubeb.cpp) code be
skipped when targeting Emscripten, and instead *_js.cpp files are used
instead (this is the cross-platform convention used by Chromium[^1], and
could be extended for other targets).
For hostevents.cpp and soundserver.cpp the entire file was replaced,
whereas for videoctrl.cpp there was enough shared logic that it was
kept, and the platform-specific bits were moved behind a Display class
that can have per-platform implementations. For cases where we need
additional private fields in the platform-specific classes, we use
a PIMPL pattern.
The *_js.cpp files with implementations are not included in this
commit, since they are closely tied to the Infinite Mac project, and
will live in its fork of DingusPPC.
[^1]: https://www.chromium.org/developers/design-documents/conventions-and-patterns-for-multi-platform-development/
2023-09-03 17:59:35 -07:00
|
|
|
impl->status = SND_STREAM_OPENED;
|
2022-07-17 05:50:32 +02:00
|
|
|
|
2020-05-15 02:36:40 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SoundServer::start_out_stream()
|
|
|
|
{
|
2024-11-09 22:25:48 -08:00
|
|
|
if (is_deterministic) {
|
|
|
|
LOG_F(9, "Starting sound output deterministic polling.");
|
|
|
|
impl->deterministic_poll_timer = TimerManager::get_instance()->add_cyclic_timer(MSECS_TO_NSECS(10), impl->deterministic_poll_cb);
|
|
|
|
return 0;
|
|
|
|
}
|
Make Emscripten build not depend on SDL2 or cubeb
While Emscripten has an SDL compabtility layer, it assumes that the
code is executing in the main browser process (and thus has access to
them DOM). The Infinite Mac project runs emulators in a worker thread
(for better performance) and has a custom API for the display, sound,
input, etc. Similarly, it does not need the cross-platform sound support
from cubeb, there there is a sound API as well.
This commit makes SDL (*_sdl.cpp) and cubeb-based (*_cubeb.cpp) code be
skipped when targeting Emscripten, and instead *_js.cpp files are used
instead (this is the cross-platform convention used by Chromium[^1], and
could be extended for other targets).
For hostevents.cpp and soundserver.cpp the entire file was replaced,
whereas for videoctrl.cpp there was enough shared logic that it was
kept, and the platform-specific bits were moved behind a Display class
that can have per-platform implementations. For cases where we need
additional private fields in the platform-specific classes, we use
a PIMPL pattern.
The *_js.cpp files with implementations are not included in this
commit, since they are closely tied to the Infinite Mac project, and
will live in its fork of DingusPPC.
[^1]: https://www.chromium.org/developers/design-documents/conventions-and-patterns-for-multi-platform-development/
2023-09-03 17:59:35 -07:00
|
|
|
return cubeb_stream_start(impl->out_stream);
|
2020-05-15 02:36:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void SoundServer::close_out_stream()
|
|
|
|
{
|
2024-11-09 22:25:48 -08:00
|
|
|
if (is_deterministic) {
|
|
|
|
LOG_F(9, "Stopping sound output deterministic polling.");
|
|
|
|
TimerManager::get_instance()->cancel_timer(impl->deterministic_poll_timer);
|
|
|
|
impl->status = SND_STREAM_CLOSED;
|
|
|
|
return;
|
|
|
|
}
|
Make Emscripten build not depend on SDL2 or cubeb
While Emscripten has an SDL compabtility layer, it assumes that the
code is executing in the main browser process (and thus has access to
them DOM). The Infinite Mac project runs emulators in a worker thread
(for better performance) and has a custom API for the display, sound,
input, etc. Similarly, it does not need the cross-platform sound support
from cubeb, there there is a sound API as well.
This commit makes SDL (*_sdl.cpp) and cubeb-based (*_cubeb.cpp) code be
skipped when targeting Emscripten, and instead *_js.cpp files are used
instead (this is the cross-platform convention used by Chromium[^1], and
could be extended for other targets).
For hostevents.cpp and soundserver.cpp the entire file was replaced,
whereas for videoctrl.cpp there was enough shared logic that it was
kept, and the platform-specific bits were moved behind a Display class
that can have per-platform implementations. For cases where we need
additional private fields in the platform-specific classes, we use
a PIMPL pattern.
The *_js.cpp files with implementations are not included in this
commit, since they are closely tied to the Infinite Mac project, and
will live in its fork of DingusPPC.
[^1]: https://www.chromium.org/developers/design-documents/conventions-and-patterns-for-multi-platform-development/
2023-09-03 17:59:35 -07:00
|
|
|
cubeb_stream_stop(impl->out_stream);
|
|
|
|
cubeb_stream_destroy(impl->out_stream);
|
|
|
|
impl->status = SND_STREAM_CLOSED;
|
2023-04-23 21:38:48 +02:00
|
|
|
LOG_F(9, "Sound output stream closed.");
|
2020-05-15 02:36:40 +02:00
|
|
|
}
|