mirror of https://github.com/bsnes-emu/bsnes.git
Update to v094r22 release.
byuu says: I fixed the hiro layout enable bug, so when you go to assign joypad input, the window disables itself so your input doesn't mess with the controls. I added "reset" to the hotkeys, in case you feel like clearing all of them at once. I added device selection support and the ability to disable audio synchronization (run > 60fps) to the ruby/OSS driver. This is exposed in tomoko's configuration file. I added checks to stringify so that assigning null char* strings to nall::string won't cause crashes anymore (technically the crash was in strlen(), which doesn't check for null strings, but whatever ... I'll do the check myself.) I hooked up BrowserDialog::folderSelect() to loading slotted media for now. Tested it by loading a Game Boy game successfully through the Super Game Boy. Definitely want to write a custom window for this though, that looks more like the library dialog. Remaining issues: - finish slotted cart loader (SGB, BSX, ST) - add DIP switch selection window (NSS) [I may end up punting this one to v096] - add more configuration panels (video, audio, timing)
This commit is contained in:
parent
99b2b4b57c
commit
7bf4cff946
|
@ -3,7 +3,7 @@
|
||||||
|
|
||||||
namespace Emulator {
|
namespace Emulator {
|
||||||
static const char Name[] = "higan";
|
static const char Name[] = "higan";
|
||||||
static const char Version[] = "094.21";
|
static const char Version[] = "094.22";
|
||||||
static const char Author[] = "byuu";
|
static const char Author[] = "byuu";
|
||||||
static const char License[] = "GPLv3";
|
static const char License[] = "GPLv3";
|
||||||
static const char Website[] = "http://byuu.org/";
|
static const char Website[] = "http://byuu.org/";
|
||||||
|
|
|
@ -31,7 +31,7 @@ auto mFixedLayout::reset() -> type& {
|
||||||
auto mFixedLayout::setEnabled(bool enabled) -> type& {
|
auto mFixedLayout::setEnabled(bool enabled) -> type& {
|
||||||
mLayout::setEnabled(enabled);
|
mLayout::setEnabled(enabled);
|
||||||
for(auto n : range(sizables())) {
|
for(auto n : range(sizables())) {
|
||||||
sizable(n)->setEnabled(sizable(n)->enabled(true));
|
sizable(n)->setEnabled(sizable(n)->enabled());
|
||||||
}
|
}
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
@ -39,7 +39,7 @@ auto mFixedLayout::setEnabled(bool enabled) -> type& {
|
||||||
auto mFixedLayout::setFont(const string& font) -> type& {
|
auto mFixedLayout::setFont(const string& font) -> type& {
|
||||||
mLayout::setFont(font);
|
mLayout::setFont(font);
|
||||||
for(auto n : range(sizables())) {
|
for(auto n : range(sizables())) {
|
||||||
sizable(n)->setFont(sizable(n)->font(true));
|
sizable(n)->setFont(sizable(n)->font());
|
||||||
}
|
}
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
@ -47,7 +47,7 @@ auto mFixedLayout::setFont(const string& font) -> type& {
|
||||||
auto mFixedLayout::setVisible(bool visible) -> type& {
|
auto mFixedLayout::setVisible(bool visible) -> type& {
|
||||||
mLayout::setVisible(visible);
|
mLayout::setVisible(visible);
|
||||||
for(auto n : range(sizables())) {
|
for(auto n : range(sizables())) {
|
||||||
sizable(n)->setVisible(sizable(n)->visible(true));
|
sizable(n)->setVisible(sizable(n)->visible());
|
||||||
}
|
}
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
|
@ -51,7 +51,7 @@ auto mHorizontalLayout::setAlignment(double alignment) -> type& {
|
||||||
auto mHorizontalLayout::setEnabled(bool enabled) -> type& {
|
auto mHorizontalLayout::setEnabled(bool enabled) -> type& {
|
||||||
mLayout::setEnabled(enabled);
|
mLayout::setEnabled(enabled);
|
||||||
for(auto n : range(sizables())) {
|
for(auto n : range(sizables())) {
|
||||||
sizable(n)->setEnabled(sizable(n)->enabled(true));
|
sizable(n)->setEnabled(sizable(n)->enabled());
|
||||||
}
|
}
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
@ -59,7 +59,7 @@ auto mHorizontalLayout::setEnabled(bool enabled) -> type& {
|
||||||
auto mHorizontalLayout::setFont(const string& font) -> type& {
|
auto mHorizontalLayout::setFont(const string& font) -> type& {
|
||||||
mLayout::setFont(font);
|
mLayout::setFont(font);
|
||||||
for(auto n : range(sizables())) {
|
for(auto n : range(sizables())) {
|
||||||
sizable(n)->setFont(sizable(n)->font(true));
|
sizable(n)->setFont(sizable(n)->font());
|
||||||
}
|
}
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
@ -121,7 +121,7 @@ auto mHorizontalLayout::setSpacing(signed spacing) -> type& {
|
||||||
auto mHorizontalLayout::setVisible(bool visible) -> type& {
|
auto mHorizontalLayout::setVisible(bool visible) -> type& {
|
||||||
mLayout::setVisible(visible);
|
mLayout::setVisible(visible);
|
||||||
for(auto n : range(sizables())) {
|
for(auto n : range(sizables())) {
|
||||||
sizable(n)->setVisible(sizable(n)->visible(true));
|
sizable(n)->setVisible(sizable(n)->visible());
|
||||||
}
|
}
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
|
@ -51,7 +51,7 @@ auto mVerticalLayout::setAlignment(double alignment) -> type& {
|
||||||
auto mVerticalLayout::setEnabled(bool enabled) -> type& {
|
auto mVerticalLayout::setEnabled(bool enabled) -> type& {
|
||||||
mLayout::setEnabled(enabled);
|
mLayout::setEnabled(enabled);
|
||||||
for(auto n : range(sizables())) {
|
for(auto n : range(sizables())) {
|
||||||
sizable(n)->setEnabled(sizable(n)->enabled(true));
|
sizable(n)->setEnabled(sizable(n)->enabled());
|
||||||
}
|
}
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
@ -59,7 +59,7 @@ auto mVerticalLayout::setEnabled(bool enabled) -> type& {
|
||||||
auto mVerticalLayout::setFont(const string& font) -> type& {
|
auto mVerticalLayout::setFont(const string& font) -> type& {
|
||||||
mLayout::setFont(font);
|
mLayout::setFont(font);
|
||||||
for(auto n : range(sizables())) {
|
for(auto n : range(sizables())) {
|
||||||
sizable(n)->setFont(sizable(n)->font(true));
|
sizable(n)->setFont(sizable(n)->font());
|
||||||
}
|
}
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
@ -123,7 +123,7 @@ auto mVerticalLayout::setSpacing(signed spacing) -> type& {
|
||||||
auto mVerticalLayout::setVisible(bool visible) -> type& {
|
auto mVerticalLayout::setVisible(bool visible) -> type& {
|
||||||
mLayout::setVisible(visible);
|
mLayout::setVisible(visible);
|
||||||
for(auto n : range(sizables())) {
|
for(auto n : range(sizables())) {
|
||||||
sizable(n)->setVisible(sizable(n)->visible(true));
|
sizable(n)->setVisible(sizable(n)->visible());
|
||||||
}
|
}
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,21 +11,12 @@ auto pLayout::destruct() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto pLayout::setEnabled(bool enabled) -> void {
|
auto pLayout::setEnabled(bool enabled) -> void {
|
||||||
for(auto& sizable : state().sizables) {
|
|
||||||
if(sizable->self()) sizable->self()->setEnabled(sizable->enabled(true));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
auto pLayout::setFont(const string& font) -> void {
|
auto pLayout::setFont(const string& font) -> void {
|
||||||
for(auto& sizable : state().sizables) {
|
|
||||||
if(sizable->self()) sizable->self()->setFont(sizable->font(true));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
auto pLayout::setVisible(bool visible) -> void {
|
auto pLayout::setVisible(bool visible) -> void {
|
||||||
for(auto& sizable : state().sizables) {
|
|
||||||
if(sizable->self()) sizable->self()->setVisible(sizable->visible(true));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -160,14 +160,14 @@ template<> struct stringify<char*> {
|
||||||
const char* _data;
|
const char* _data;
|
||||||
auto data() const -> const char* { return _data; }
|
auto data() const -> const char* { return _data; }
|
||||||
auto size() const -> unsigned { return strlen(_data); }
|
auto size() const -> unsigned { return strlen(_data); }
|
||||||
stringify(char* source) : _data(source) {}
|
stringify(char* source) : _data(source ? source : "") {}
|
||||||
};
|
};
|
||||||
|
|
||||||
template<> struct stringify<const char*> {
|
template<> struct stringify<const char*> {
|
||||||
const char* _data;
|
const char* _data;
|
||||||
auto data() const -> const char* { return _data; }
|
auto data() const -> const char* { return _data; }
|
||||||
auto size() const -> unsigned { return strlen(_data); }
|
auto size() const -> unsigned { return strlen(_data); }
|
||||||
stringify(const char* source) : _data(source) {}
|
stringify(const char* source) : _data(source ? source : "") {}
|
||||||
};
|
};
|
||||||
|
|
||||||
//strings
|
//strings
|
||||||
|
|
|
@ -1,18 +1,19 @@
|
||||||
struct Audio {
|
struct Audio {
|
||||||
static const char* Handle;
|
static const nall::string Device;
|
||||||
static const char* Synchronize;
|
static const nall::string Handle;
|
||||||
static const char* Frequency;
|
static const nall::string Synchronize;
|
||||||
static const char* Latency;
|
static const nall::string Frequency;
|
||||||
|
static const nall::string Latency;
|
||||||
|
|
||||||
virtual bool cap(const nall::string& name) { return false; }
|
virtual ~Audio() = default;
|
||||||
virtual nall::any get(const nall::string& name) { return false; }
|
|
||||||
virtual bool set(const nall::string& name, const nall::any& value) { return false; }
|
|
||||||
|
|
||||||
virtual void sample(uint16_t left, uint16_t right) {}
|
virtual auto cap(const nall::string& name) -> bool { return false; }
|
||||||
virtual void clear() {}
|
virtual auto get(const nall::string& name) -> nall::any { return false; }
|
||||||
virtual bool init() { return true; }
|
virtual auto set(const nall::string& name, const nall::any& value) -> bool { return false; }
|
||||||
virtual void term() {}
|
|
||||||
|
|
||||||
Audio() {}
|
virtual auto sample(uint16_t left, uint16_t right) -> void {}
|
||||||
virtual ~Audio() {}
|
virtual auto clear() -> void {}
|
||||||
|
|
||||||
|
virtual auto init() -> bool { return true; }
|
||||||
|
virtual auto term() -> void {}
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,9 +1,3 @@
|
||||||
/*
|
|
||||||
audio.openal (2007-12-26)
|
|
||||||
author: Nach
|
|
||||||
contributors: byuu, wertigon, _willow_
|
|
||||||
*/
|
|
||||||
|
|
||||||
#if defined(PLATFORM_MACOSX)
|
#if defined(PLATFORM_MACOSX)
|
||||||
#include <OpenAL/al.h>
|
#include <OpenAL/al.h>
|
||||||
#include <OpenAL/alc.h>
|
#include <OpenAL/alc.h>
|
||||||
|
@ -14,15 +8,14 @@
|
||||||
|
|
||||||
namespace ruby {
|
namespace ruby {
|
||||||
|
|
||||||
class pAudioOpenAL {
|
struct pAudioOpenAL {
|
||||||
public:
|
|
||||||
struct {
|
struct {
|
||||||
ALCdevice* handle;
|
ALCdevice* handle;
|
||||||
ALCcontext* context;
|
ALCcontext* context;
|
||||||
ALuint source;
|
ALuint source;
|
||||||
ALenum format;
|
ALenum format;
|
||||||
unsigned latency;
|
unsigned latency;
|
||||||
unsigned queue_length;
|
unsigned queueLength;
|
||||||
} device;
|
} device;
|
||||||
|
|
||||||
struct {
|
struct {
|
||||||
|
@ -37,21 +30,21 @@ public:
|
||||||
unsigned latency;
|
unsigned latency;
|
||||||
} settings;
|
} settings;
|
||||||
|
|
||||||
bool cap(const string& name) {
|
auto cap(const string& name) -> bool {
|
||||||
if(name == Audio::Synchronize) return true;
|
if(name == Audio::Synchronize) return true;
|
||||||
if(name == Audio::Frequency) return true;
|
if(name == Audio::Frequency) return true;
|
||||||
if(name == Audio::Latency) return true;
|
if(name == Audio::Latency) return true;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
any get(const string& name) {
|
auto get(const string& name) -> any {
|
||||||
if(name == Audio::Synchronize) return settings.synchronize;
|
if(name == Audio::Synchronize) return settings.synchronize;
|
||||||
if(name == Audio::Frequency) return settings.frequency;
|
if(name == Audio::Frequency) return settings.frequency;
|
||||||
if(name == Audio::Latency) return settings.latency;
|
if(name == Audio::Latency) return settings.latency;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool set(const string& name, const any& value) {
|
auto set(const string& name, const any& value) -> bool {
|
||||||
if(name == Audio::Synchronize) {
|
if(name == Audio::Synchronize) {
|
||||||
settings.synchronize = any_cast<bool>(value);
|
settings.synchronize = any_cast<bool>(value);
|
||||||
return true;
|
return true;
|
||||||
|
@ -65,7 +58,7 @@ public:
|
||||||
if(name == Audio::Latency) {
|
if(name == Audio::Latency) {
|
||||||
if(settings.latency != any_cast<unsigned>(value)) {
|
if(settings.latency != any_cast<unsigned>(value)) {
|
||||||
settings.latency = any_cast<unsigned>(value);
|
settings.latency = any_cast<unsigned>(value);
|
||||||
update_latency();
|
updateLatency();
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -73,7 +66,7 @@ public:
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void sample(uint16_t sl, uint16_t sr) {
|
auto sample(uint16_t sl, uint16_t sr) -> void {
|
||||||
buffer.data[buffer.length++] = sl + (sr << 16);
|
buffer.data[buffer.length++] = sl + (sr << 16);
|
||||||
if(buffer.length < buffer.size) return;
|
if(buffer.length < buffer.size) return;
|
||||||
|
|
||||||
|
@ -84,17 +77,17 @@ public:
|
||||||
while(processed--) {
|
while(processed--) {
|
||||||
alSourceUnqueueBuffers(device.source, 1, &albuffer);
|
alSourceUnqueueBuffers(device.source, 1, &albuffer);
|
||||||
alDeleteBuffers(1, &albuffer);
|
alDeleteBuffers(1, &albuffer);
|
||||||
device.queue_length--;
|
device.queueLength--;
|
||||||
}
|
}
|
||||||
//wait for buffer playback to catch up to sample generation if not synchronizing
|
//wait for buffer playback to catch up to sample generation if not synchronizing
|
||||||
if(settings.synchronize == false || device.queue_length < 3) break;
|
if(settings.synchronize == false || device.queueLength < 3) break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(device.queue_length < 3) {
|
if(device.queueLength < 3) {
|
||||||
alGenBuffers(1, &albuffer);
|
alGenBuffers(1, &albuffer);
|
||||||
alBufferData(albuffer, device.format, buffer.data, buffer.size * 4, settings.frequency);
|
alBufferData(albuffer, device.format, buffer.data, buffer.size * 4, settings.frequency);
|
||||||
alSourceQueueBuffers(device.source, 1, &albuffer);
|
alSourceQueueBuffers(device.source, 1, &albuffer);
|
||||||
device.queue_length++;
|
device.queueLength++;
|
||||||
}
|
}
|
||||||
|
|
||||||
ALint playing;
|
ALint playing;
|
||||||
|
@ -103,22 +96,16 @@ public:
|
||||||
buffer.length = 0;
|
buffer.length = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void clear() {
|
auto clear() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
void update_latency() {
|
auto init() -> bool {
|
||||||
if(buffer.data) delete[] buffer.data;
|
updateLatency();
|
||||||
buffer.size = settings.frequency * settings.latency / 1000.0 + 0.5;
|
device.queueLength = 0;
|
||||||
buffer.data = new uint32_t[buffer.size];
|
|
||||||
}
|
|
||||||
|
|
||||||
bool init() {
|
|
||||||
update_latency();
|
|
||||||
device.queue_length = 0;
|
|
||||||
|
|
||||||
bool success = false;
|
bool success = false;
|
||||||
if(device.handle = alcOpenDevice(NULL)) {
|
if(device.handle = alcOpenDevice(nullptr)) {
|
||||||
if(device.context = alcCreateContext(device.handle, NULL)) {
|
if(device.context = alcCreateContext(device.handle, nullptr)) {
|
||||||
alcMakeContextCurrent(device.context);
|
alcMakeContextCurrent(device.context);
|
||||||
alGenSources(1, &device.source);
|
alGenSources(1, &device.source);
|
||||||
|
|
||||||
|
@ -147,7 +134,7 @@ public:
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void term() {
|
auto term() -> void {
|
||||||
if(alIsSource(device.source) == AL_TRUE) {
|
if(alIsSource(device.source) == AL_TRUE) {
|
||||||
int playing = 0;
|
int playing = 0;
|
||||||
alGetSourcei(device.source, AL_SOURCE_STATE, &playing);
|
alGetSourcei(device.source, AL_SOURCE_STATE, &playing);
|
||||||
|
@ -159,7 +146,7 @@ public:
|
||||||
ALuint albuffer = 0;
|
ALuint albuffer = 0;
|
||||||
alSourceUnqueueBuffers(device.source, 1, &albuffer);
|
alSourceUnqueueBuffers(device.source, 1, &albuffer);
|
||||||
alDeleteBuffers(1, &albuffer);
|
alDeleteBuffers(1, &albuffer);
|
||||||
device.queue_length--;
|
device.queueLength--;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -168,7 +155,7 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
if(device.context) {
|
if(device.context) {
|
||||||
alcMakeContextCurrent(NULL);
|
alcMakeContextCurrent(nullptr);
|
||||||
alcDestroyContext(device.context);
|
alcDestroyContext(device.context);
|
||||||
device.context = 0;
|
device.context = 0;
|
||||||
}
|
}
|
||||||
|
@ -189,7 +176,7 @@ public:
|
||||||
device.handle = 0;
|
device.handle = 0;
|
||||||
device.context = 0;
|
device.context = 0;
|
||||||
device.format = AL_FORMAT_STEREO16;
|
device.format = AL_FORMAT_STEREO16;
|
||||||
device.queue_length = 0;
|
device.queueLength = 0;
|
||||||
|
|
||||||
buffer.data = 0;
|
buffer.data = 0;
|
||||||
buffer.length = 0;
|
buffer.length = 0;
|
||||||
|
@ -203,6 +190,27 @@ public:
|
||||||
~pAudioOpenAL() {
|
~pAudioOpenAL() {
|
||||||
term();
|
term();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
auto queryDevices() -> lstring {
|
||||||
|
lstring result;
|
||||||
|
|
||||||
|
const char* buffer = alcGetString(nullptr, ALC_DEVICE_SPECIFIER);
|
||||||
|
if(!buffer) return result;
|
||||||
|
|
||||||
|
while(buffer[0] || buffer[1]) {
|
||||||
|
result.append(buffer);
|
||||||
|
while(buffer[0]) buffer++;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto updateLatency() -> void {
|
||||||
|
if(buffer.data) delete[] buffer.data;
|
||||||
|
buffer.size = settings.frequency * settings.latency / 1000.0 + 0.5;
|
||||||
|
buffer.data = new uint32_t[buffer.size]();
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
DeclareAudio(OpenAL)
|
DeclareAudio(OpenAL)
|
||||||
|
|
|
@ -1,8 +1,3 @@
|
||||||
/*
|
|
||||||
audio.oss (2007-12-26)
|
|
||||||
author: Nach
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
#include <sys/ioctl.h>
|
#include <sys/ioctl.h>
|
||||||
|
@ -16,95 +11,112 @@
|
||||||
//Failing that, one can disable OSS4 ioctl calls inside init() and remove the below defines
|
//Failing that, one can disable OSS4 ioctl calls inside init() and remove the below defines
|
||||||
|
|
||||||
#ifndef SNDCTL_DSP_COOKEDMODE
|
#ifndef SNDCTL_DSP_COOKEDMODE
|
||||||
#define SNDCTL_DSP_COOKEDMODE _IOW('P', 30, int)
|
#define SNDCTL_DSP_COOKEDMODE _IOW('P', 30, signed)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifndef SNDCTL_DSP_POLICY
|
#ifndef SNDCTL_DSP_POLICY
|
||||||
#define SNDCTL_DSP_POLICY _IOW('P', 45, int)
|
#define SNDCTL_DSP_POLICY _IOW('P', 45, signed)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
namespace ruby {
|
namespace ruby {
|
||||||
|
|
||||||
class pAudioOSS {
|
struct pAudioOSS {
|
||||||
public:
|
|
||||||
struct {
|
struct {
|
||||||
int fd;
|
signed fd = -1;
|
||||||
int format;
|
signed format = AFMT_S16_LE;
|
||||||
int channels;
|
signed channels = 2;
|
||||||
const char* name;
|
|
||||||
} device;
|
} device;
|
||||||
|
|
||||||
struct {
|
struct {
|
||||||
unsigned frequency;
|
string device = "/dev/dsp";
|
||||||
|
bool synchronize = true;
|
||||||
|
unsigned frequency = 22050;
|
||||||
} settings;
|
} settings;
|
||||||
|
|
||||||
bool cap(const string& name) {
|
~pAudioOSS() {
|
||||||
|
term();
|
||||||
|
}
|
||||||
|
|
||||||
|
auto cap(const string& name) -> bool {
|
||||||
|
if(name == Audio::Device) return true;
|
||||||
|
if(name == Audio::Synchronize) return true;
|
||||||
if(name == Audio::Frequency) return true;
|
if(name == Audio::Frequency) return true;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
any get(const string& name) {
|
auto get(const string& name) -> any {
|
||||||
|
if(name == Audio::Device) return settings.device;
|
||||||
|
if(name == Audio::Synchronize) return settings.synchronize;
|
||||||
if(name == Audio::Frequency) return settings.frequency;
|
if(name == Audio::Frequency) return settings.frequency;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool set(const string& name, const any& value) {
|
auto set(const string& name, const any& value) -> bool {
|
||||||
|
if(name == Audio::Device) {
|
||||||
|
settings.device = any_cast<string>(value);
|
||||||
|
if(!settings.device) settings.device = "/dev/dsp";
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(name == Audio::Synchronize) {
|
||||||
|
settings.synchronize = any_cast<bool>(value);
|
||||||
|
updateSynchronization();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
if(name == Audio::Frequency) {
|
if(name == Audio::Frequency) {
|
||||||
settings.frequency = any_cast<unsigned>(value);
|
settings.frequency = any_cast<unsigned>(value);
|
||||||
if(device.fd > 0) init();
|
if(device.fd >= 0) init();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void sample(uint16_t sl, uint16_t sr) {
|
auto sample(uint16_t left, uint16_t right) -> void {
|
||||||
uint32_t sample = sl + (sr << 16);
|
uint32_t sample = left << 0 | right << 16;
|
||||||
unsigned unused = write(device.fd, &sample, 4);
|
auto unused = write(device.fd, &sample, 4);
|
||||||
}
|
}
|
||||||
|
|
||||||
void clear() {
|
auto clear() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
bool init() {
|
auto init() -> bool {
|
||||||
term();
|
term();
|
||||||
|
|
||||||
device.fd = open(device.name, O_WRONLY, O_NONBLOCK);
|
device.fd = open(settings.device, O_WRONLY, O_NONBLOCK);
|
||||||
if(device.fd < 0) return false;
|
if(device.fd < 0) return false;
|
||||||
|
|
||||||
#if 1 //SOUND_VERSION >= 0x040000
|
#if 1 //SOUND_VERSION >= 0x040000
|
||||||
//attempt to enable OSS4-specific features regardless of version
|
//attempt to enable OSS4-specific features regardless of version
|
||||||
//OSS3 ioctl calls will silently fail, but sound will still work
|
//OSS3 ioctl calls will silently fail, but sound will still work
|
||||||
int cooked = 1, policy = 4; //policy should be 0 - 10, lower = less latency, more CPU usage
|
signed cooked = 1, policy = 4; //policy should be 0 - 10, lower = less latency, more CPU usage
|
||||||
ioctl(device.fd, SNDCTL_DSP_COOKEDMODE, &cooked);
|
ioctl(device.fd, SNDCTL_DSP_COOKEDMODE, &cooked);
|
||||||
ioctl(device.fd, SNDCTL_DSP_POLICY, &policy);
|
ioctl(device.fd, SNDCTL_DSP_POLICY, &policy);
|
||||||
#endif
|
#endif
|
||||||
int freq = settings.frequency;
|
signed freq = settings.frequency;
|
||||||
ioctl(device.fd, SNDCTL_DSP_CHANNELS, &device.channels);
|
ioctl(device.fd, SNDCTL_DSP_CHANNELS, &device.channels);
|
||||||
ioctl(device.fd, SNDCTL_DSP_SETFMT, &device.format);
|
ioctl(device.fd, SNDCTL_DSP_SETFMT, &device.format);
|
||||||
ioctl(device.fd, SNDCTL_DSP_SPEED, &freq);
|
ioctl(device.fd, SNDCTL_DSP_SPEED, &freq);
|
||||||
|
|
||||||
|
updateSynchronization();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void term() {
|
auto term() -> void {
|
||||||
if(device.fd > 0) {
|
if(device.fd >= 0) {
|
||||||
close(device.fd);
|
close(device.fd);
|
||||||
device.fd = -1;
|
device.fd = -1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pAudioOSS() {
|
private:
|
||||||
device.fd = -1;
|
auto updateSynchronization() -> void {
|
||||||
device.format = AFMT_S16_LE;
|
if(device.fd < 0) return;
|
||||||
device.channels = 2;
|
auto flags = fcntl(device.fd, F_GETFL);
|
||||||
device.name = "/dev/dsp";
|
if(flags < 0) return;
|
||||||
|
settings.synchronize ? flags &=~ O_NONBLOCK : flags |= O_NONBLOCK;
|
||||||
settings.frequency = 22050;
|
fcntl(device.fd, F_SETFL, flags);
|
||||||
}
|
|
||||||
|
|
||||||
~pAudioOSS() {
|
|
||||||
term();
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -19,21 +19,21 @@ using namespace nall;
|
||||||
|
|
||||||
#define DeclareVideo(Name) \
|
#define DeclareVideo(Name) \
|
||||||
struct Video##Name : Video { \
|
struct Video##Name : Video { \
|
||||||
bool cap(const string& name) { return p.cap(name); } \
|
|
||||||
any get(const string& name) { return p.get(name); } \
|
|
||||||
bool set(const string& name, const any& value) { return p.set(name, value); } \
|
|
||||||
\
|
|
||||||
bool lock(uint32_t*& data, unsigned& pitch, unsigned width, unsigned height) { return p.lock(data, pitch, width, height); } \
|
|
||||||
void unlock() { p.unlock(); } \
|
|
||||||
\
|
|
||||||
void clear() { p.clear(); } \
|
|
||||||
void refresh() { p.refresh(); } \
|
|
||||||
bool init() { return p.init(); } \
|
|
||||||
void term() { p.term(); } \
|
|
||||||
\
|
|
||||||
Video##Name() : p(*new pVideo##Name) {} \
|
Video##Name() : p(*new pVideo##Name) {} \
|
||||||
~Video##Name() { delete &p; } \
|
~Video##Name() { delete &p; } \
|
||||||
\
|
\
|
||||||
|
auto cap(const string& name) -> bool { return p.cap(name); } \
|
||||||
|
auto get(const string& name) -> any { return p.get(name); } \
|
||||||
|
auto set(const string& name, const any& value) -> bool { return p.set(name, value); } \
|
||||||
|
\
|
||||||
|
auto lock(uint32_t*& data, unsigned& pitch, unsigned width, unsigned height) -> bool { return p.lock(data, pitch, width, height); } \
|
||||||
|
auto unlock() -> void { p.unlock(); } \
|
||||||
|
auto clear() -> void { p.clear(); } \
|
||||||
|
auto refresh() -> void { p.refresh(); } \
|
||||||
|
\
|
||||||
|
auto init() -> bool { return p.init(); } \
|
||||||
|
auto term() -> void { p.term(); } \
|
||||||
|
\
|
||||||
private: \
|
private: \
|
||||||
pVideo##Name& p; \
|
pVideo##Name& p; \
|
||||||
};
|
};
|
||||||
|
@ -78,18 +78,19 @@ using namespace nall;
|
||||||
|
|
||||||
#define DeclareAudio(Name) \
|
#define DeclareAudio(Name) \
|
||||||
struct Audio##Name : Audio { \
|
struct Audio##Name : Audio { \
|
||||||
bool cap(const string& name) { return p.cap(name); } \
|
|
||||||
any get(const string& name) { return p.get(name); } \
|
|
||||||
bool set(const string& name, const any& value) { return p.set(name, value); } \
|
|
||||||
\
|
|
||||||
void sample(uint16_t left, uint16_t right) { p.sample(left, right); } \
|
|
||||||
void clear() { p.clear(); } \
|
|
||||||
bool init() { return p.init(); } \
|
|
||||||
void term() { p.term(); } \
|
|
||||||
\
|
|
||||||
Audio##Name() : p(*new pAudio##Name) {} \
|
Audio##Name() : p(*new pAudio##Name) {} \
|
||||||
~Audio##Name() { delete &p; } \
|
~Audio##Name() { delete &p; } \
|
||||||
\
|
\
|
||||||
|
auto cap(const string& name) -> bool { return p.cap(name); } \
|
||||||
|
auto get(const string& name) -> any { return p.get(name); } \
|
||||||
|
auto set(const string& name, const any& value) -> bool { return p.set(name, value); } \
|
||||||
|
\
|
||||||
|
auto sample(uint16_t left, uint16_t right) -> void { p.sample(left, right); } \
|
||||||
|
auto clear() -> void { p.clear(); } \
|
||||||
|
\
|
||||||
|
auto init() -> bool { return p.init(); } \
|
||||||
|
auto term() -> void { p.term(); } \
|
||||||
|
\
|
||||||
private: \
|
private: \
|
||||||
pAudio##Name& p; \
|
pAudio##Name& p; \
|
||||||
};
|
};
|
||||||
|
@ -140,9 +141,9 @@ using namespace nall;
|
||||||
auto acquire() -> bool { return p.acquire(); } \
|
auto acquire() -> bool { return p.acquire(); } \
|
||||||
auto unacquire() -> bool { return p.unacquire(); } \
|
auto unacquire() -> bool { return p.unacquire(); } \
|
||||||
auto acquired() -> bool { return p.acquired(); } \
|
auto acquired() -> bool { return p.acquired(); } \
|
||||||
\
|
|
||||||
auto poll() -> vector<shared_pointer<HID::Device>> { return p.poll(); } \
|
auto poll() -> vector<shared_pointer<HID::Device>> { return p.poll(); } \
|
||||||
auto rumble(uint64_t id, bool enable) -> bool { return p.rumble(id, enable); } \
|
auto rumble(uint64_t id, bool enable) -> bool { return p.rumble(id, enable); } \
|
||||||
|
\
|
||||||
auto init() -> bool { return p.init(); } \
|
auto init() -> bool { return p.init(); } \
|
||||||
auto term() -> void { p.term(); } \
|
auto term() -> void { p.term(); } \
|
||||||
\
|
\
|
||||||
|
|
|
@ -5,7 +5,6 @@ struct Input {
|
||||||
static const nall::string JoypadSupport;
|
static const nall::string JoypadSupport;
|
||||||
static const nall::string JoypadRumbleSupport;
|
static const nall::string JoypadRumbleSupport;
|
||||||
|
|
||||||
Input() = default;
|
|
||||||
virtual ~Input() = default;
|
virtual ~Input() = default;
|
||||||
|
|
||||||
virtual auto cap(const nall::string& name) -> bool { return false; }
|
virtual auto cap(const nall::string& name) -> bool { return false; }
|
||||||
|
@ -15,9 +14,9 @@ struct Input {
|
||||||
virtual auto acquire() -> bool { return false; }
|
virtual auto acquire() -> bool { return false; }
|
||||||
virtual auto unacquire() -> bool { return false; }
|
virtual auto unacquire() -> bool { return false; }
|
||||||
virtual auto acquired() -> bool { return false; }
|
virtual auto acquired() -> bool { return false; }
|
||||||
|
|
||||||
virtual auto poll() -> nall::vector<nall::shared_pointer<nall::HID::Device>> { return {}; }
|
virtual auto poll() -> nall::vector<nall::shared_pointer<nall::HID::Device>> { return {}; }
|
||||||
virtual auto rumble(uint64_t id, bool enable) -> bool { return false; }
|
virtual auto rumble(uint64_t id, bool enable) -> bool { return false; }
|
||||||
|
|
||||||
virtual auto init() -> bool { return true; }
|
virtual auto init() -> bool { return true; }
|
||||||
virtual auto term() -> void {}
|
virtual auto term() -> void {}
|
||||||
};
|
};
|
||||||
|
|
111
ruby/ruby.cpp
111
ruby/ruby.cpp
|
@ -13,70 +13,70 @@ InputInterface input;
|
||||||
|
|
||||||
/* VideoInterface */
|
/* VideoInterface */
|
||||||
|
|
||||||
const char* Video::Handle = "Handle";
|
const string Video::Handle = "Handle";
|
||||||
const char* Video::Synchronize = "Synchronize";
|
const string Video::Synchronize = "Synchronize";
|
||||||
const char* Video::Depth = "Depth";
|
const string Video::Depth = "Depth";
|
||||||
const char* Video::Filter = "Filter";
|
const string Video::Filter = "Filter";
|
||||||
const char* Video::Shader = "Shader";
|
const string Video::Shader = "Shader";
|
||||||
|
|
||||||
const unsigned Video::FilterNearest = 0;
|
const unsigned Video::FilterNearest = 0;
|
||||||
const unsigned Video::FilterLinear = 1;
|
const unsigned Video::FilterLinear = 1;
|
||||||
|
|
||||||
void VideoInterface::driver(const char* driver) {
|
auto VideoInterface::driver(string driver) -> void {
|
||||||
if(p) term();
|
if(p) term();
|
||||||
|
|
||||||
if(!driver || !*driver) driver = optimalDriver();
|
if(!driver) driver = optimalDriver();
|
||||||
|
|
||||||
if(0);
|
if(0);
|
||||||
|
|
||||||
#ifdef VIDEO_CGL
|
#ifdef VIDEO_CGL
|
||||||
else if(!strcmp(driver, "OpenGL")) p = new VideoCGL();
|
else if(driver == "OpenGL") p = new VideoCGL();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef VIDEO_DIRECT3D
|
#ifdef VIDEO_DIRECT3D
|
||||||
else if(!strcmp(driver, "Direct3D")) p = new VideoD3D();
|
else if(driver == "Direct3D") p = new VideoD3D();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef VIDEO_DIRECTDRAW
|
#ifdef VIDEO_DIRECTDRAW
|
||||||
else if(!strcmp(driver, "DirectDraw")) p = new VideoDD();
|
else if(driver == "DirectDraw") p = new VideoDD();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef VIDEO_GDI
|
#ifdef VIDEO_GDI
|
||||||
else if(!strcmp(driver, "GDI")) p = new VideoGDI();
|
else if(driver == "GDI") p = new VideoGDI();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef VIDEO_GLX
|
#ifdef VIDEO_GLX
|
||||||
else if(!strcmp(driver, "OpenGL")) p = new VideoGLX();
|
else if(driver == "OpenGL") p = new VideoGLX();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef VIDEO_QTOPENGL
|
#ifdef VIDEO_QTOPENGL
|
||||||
else if(!strcmp(driver, "Qt-OpenGL")) p = new VideoQtOpenGL();
|
else if(driver == "Qt-OpenGL") p = new VideoQtOpenGL();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef VIDEO_QTRASTER
|
#ifdef VIDEO_QTRASTER
|
||||||
else if(!strcmp(driver, "Qt-Raster")) p = new VideoQtRaster();
|
else if(driver == "Qt-Raster") p = new VideoQtRaster();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef VIDEO_SDL
|
#ifdef VIDEO_SDL
|
||||||
else if(!strcmp(driver, "SDL")) p = new VideoSDL();
|
else if(driver == "SDL") p = new VideoSDL();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef VIDEO_WGL
|
#ifdef VIDEO_WGL
|
||||||
else if(!strcmp(driver, "OpenGL")) p = new VideoWGL();
|
else if(driver == "OpenGL") p = new VideoWGL();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef VIDEO_XSHM
|
#ifdef VIDEO_XSHM
|
||||||
else if(!strcmp(driver, "XShm")) p = new VideoXShm();
|
else if(driver == "XShm") p = new VideoXShm();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef VIDEO_XV
|
#ifdef VIDEO_XV
|
||||||
else if(!strcmp(driver, "X-Video")) p = new VideoXv();
|
else if(driver == "X-Video") p = new VideoXv();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
else p = new Video();
|
else p = new Video();
|
||||||
}
|
}
|
||||||
|
|
||||||
const char* VideoInterface::optimalDriver() {
|
auto VideoInterface::optimalDriver() -> string {
|
||||||
#if defined(VIDEO_WGL)
|
#if defined(VIDEO_WGL)
|
||||||
return "OpenGL";
|
return "OpenGL";
|
||||||
#elif defined(VIDEO_DIRECT3D)
|
#elif defined(VIDEO_DIRECT3D)
|
||||||
|
@ -103,7 +103,7 @@ const char* VideoInterface::optimalDriver() {
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
const char* VideoInterface::safestDriver() {
|
auto VideoInterface::safestDriver() -> string {
|
||||||
#if defined(VIDEO_DIRECT3D)
|
#if defined(VIDEO_DIRECT3D)
|
||||||
return "Direct3D";
|
return "Direct3D";
|
||||||
#elif defined(VIDEO_WGL)
|
#elif defined(VIDEO_WGL)
|
||||||
|
@ -130,7 +130,7 @@ const char* VideoInterface::safestDriver() {
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
const char* VideoInterface::availableDrivers() {
|
auto VideoInterface::availableDrivers() -> string {
|
||||||
return
|
return
|
||||||
|
|
||||||
//Windows
|
//Windows
|
||||||
|
@ -178,12 +178,12 @@ const char* VideoInterface::availableDrivers() {
|
||||||
"None";
|
"None";
|
||||||
}
|
}
|
||||||
|
|
||||||
bool VideoInterface::init() {
|
auto VideoInterface::init() -> bool {
|
||||||
if(!p) driver();
|
if(!p) driver();
|
||||||
return p->init();
|
return p->init();
|
||||||
}
|
}
|
||||||
|
|
||||||
void VideoInterface::term() {
|
auto VideoInterface::term() -> void {
|
||||||
if(p) {
|
if(p) {
|
||||||
p->term();
|
p->term();
|
||||||
delete p;
|
delete p;
|
||||||
|
@ -191,66 +191,66 @@ void VideoInterface::term() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool VideoInterface::cap(const string& name) { return p ? p->cap(name) : false; }
|
|
||||||
any VideoInterface::get(const string& name) { return p ? p->get(name) : false; }
|
|
||||||
bool VideoInterface::set(const string& name, const any& value) { return p ? p->set(name, value) : false; }
|
|
||||||
bool VideoInterface::lock(uint32_t*& data, unsigned& pitch, unsigned width, unsigned height) { return p ? p->lock(data, pitch, width, height) : false; }
|
|
||||||
void VideoInterface::unlock() { if(p) p->unlock(); }
|
|
||||||
void VideoInterface::clear() { if(p) p->clear(); }
|
|
||||||
void VideoInterface::refresh() { if(p) p->refresh(); }
|
|
||||||
VideoInterface::VideoInterface() : p(nullptr) {}
|
|
||||||
VideoInterface::~VideoInterface() { term(); }
|
VideoInterface::~VideoInterface() { term(); }
|
||||||
|
auto VideoInterface::cap(const string& name) -> bool { return p ? p->cap(name) : false; }
|
||||||
|
auto VideoInterface::get(const string& name) -> any { return p ? p->get(name) : false; }
|
||||||
|
auto VideoInterface::set(const string& name, const any& value) -> bool { return p ? p->set(name, value) : false; }
|
||||||
|
auto VideoInterface::lock(uint32_t*& data, unsigned& pitch, unsigned width, unsigned height) -> bool { return p ? p->lock(data, pitch, width, height) : false; }
|
||||||
|
auto VideoInterface::unlock() -> void { if(p) p->unlock(); }
|
||||||
|
auto VideoInterface::clear() -> void { if(p) p->clear(); }
|
||||||
|
auto VideoInterface::refresh() -> void { if(p) p->refresh(); }
|
||||||
|
|
||||||
/* AudioInterface */
|
/* AudioInterface */
|
||||||
|
|
||||||
const char* Audio::Handle = "Handle";
|
const string Audio::Device = "Device";
|
||||||
const char* Audio::Synchronize = "Synchronize";
|
const string Audio::Handle = "Handle";
|
||||||
const char* Audio::Frequency = "Frequency";
|
const string Audio::Synchronize = "Synchronize";
|
||||||
const char* Audio::Latency = "Latency";
|
const string Audio::Frequency = "Frequency";
|
||||||
|
const string Audio::Latency = "Latency";
|
||||||
|
|
||||||
void AudioInterface::driver(const char* driver) {
|
auto AudioInterface::driver(string driver) -> void {
|
||||||
if(p) term();
|
if(p) term();
|
||||||
|
|
||||||
if(!driver || !*driver) driver = optimalDriver();
|
if(!driver) driver = optimalDriver();
|
||||||
|
|
||||||
if(0);
|
if(0);
|
||||||
|
|
||||||
#ifdef AUDIO_ALSA
|
#ifdef AUDIO_ALSA
|
||||||
else if(!strcmp(driver, "ALSA")) p = new AudioALSA();
|
else if(driver == "ALSA") p = new AudioALSA();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef AUDIO_AO
|
#ifdef AUDIO_AO
|
||||||
else if(!strcmp(driver, "libao")) p = new AudioAO();
|
else if(driver == "libao") p = new AudioAO();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef AUDIO_DIRECTSOUND
|
#ifdef AUDIO_DIRECTSOUND
|
||||||
else if(!strcmp(driver, "DirectSound")) p = new AudioDS();
|
else if(driver == "DirectSound") p = new AudioDS();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef AUDIO_OPENAL
|
#ifdef AUDIO_OPENAL
|
||||||
else if(!strcmp(driver, "OpenAL")) p = new AudioOpenAL();
|
else if(driver == "OpenAL") p = new AudioOpenAL();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef AUDIO_OSS
|
#ifdef AUDIO_OSS
|
||||||
else if(!strcmp(driver, "OSS")) p = new AudioOSS();
|
else if(driver == "OSS") p = new AudioOSS();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef AUDIO_PULSEAUDIO
|
#ifdef AUDIO_PULSEAUDIO
|
||||||
else if(!strcmp(driver, "PulseAudio")) p = new AudioPulseAudio();
|
else if(driver == "PulseAudio") p = new AudioPulseAudio();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef AUDIO_PULSEAUDIOSIMPLE
|
#ifdef AUDIO_PULSEAUDIOSIMPLE
|
||||||
else if(!strcmp(driver, "PulseAudioSimple")) p = new AudioPulseAudioSimple();
|
else if(driver == "PulseAudioSimple") p = new AudioPulseAudioSimple();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef AUDIO_XAUDIO2
|
#ifdef AUDIO_XAUDIO2
|
||||||
else if(!strcmp(driver, "XAudio2")) p = new AudioXAudio2();
|
else if(driver == "XAudio2") p = new AudioXAudio2();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
else p = new Audio();
|
else p = new Audio();
|
||||||
}
|
}
|
||||||
|
|
||||||
const char* AudioInterface::optimalDriver() {
|
auto AudioInterface::optimalDriver() -> string {
|
||||||
#if defined(AUDIO_XAUDIO2)
|
#if defined(AUDIO_XAUDIO2)
|
||||||
return "XAudio2";
|
return "XAudio2";
|
||||||
#elif defined(AUDIO_DIRECTSOUND)
|
#elif defined(AUDIO_DIRECTSOUND)
|
||||||
|
@ -274,7 +274,7 @@ const char* AudioInterface::optimalDriver() {
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
const char* AudioInterface::safestDriver() {
|
auto AudioInterface::safestDriver() -> string {
|
||||||
#if defined(AUDIO_DIRECTSOUND)
|
#if defined(AUDIO_DIRECTSOUND)
|
||||||
return "DirectSound";
|
return "DirectSound";
|
||||||
#elif defined(AUDIO_XAUDIO2)
|
#elif defined(AUDIO_XAUDIO2)
|
||||||
|
@ -298,7 +298,7 @@ const char* AudioInterface::safestDriver() {
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
const char* AudioInterface::availableDrivers() {
|
auto AudioInterface::availableDrivers() -> string {
|
||||||
return
|
return
|
||||||
|
|
||||||
//Windows
|
//Windows
|
||||||
|
@ -340,12 +340,12 @@ const char* AudioInterface::availableDrivers() {
|
||||||
"None";
|
"None";
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AudioInterface::init() {
|
auto AudioInterface::init() -> bool {
|
||||||
if(!p) driver();
|
if(!p) driver();
|
||||||
return p->init();
|
return p->init();
|
||||||
}
|
}
|
||||||
|
|
||||||
void AudioInterface::term() {
|
auto AudioInterface::term() -> void {
|
||||||
if(p) {
|
if(p) {
|
||||||
p->term();
|
p->term();
|
||||||
delete p;
|
delete p;
|
||||||
|
@ -353,13 +353,12 @@ void AudioInterface::term() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AudioInterface::cap(const string& name) { return p ? p->cap(name) : false; }
|
|
||||||
any AudioInterface::get(const string& name) { return p ? p->get(name) : false; }
|
|
||||||
bool AudioInterface::set(const string& name, const any& value) { return p ? p->set(name, value) : false; }
|
|
||||||
void AudioInterface::sample(uint16_t left, uint16_t right) { if(p) p->sample(left, right); }
|
|
||||||
void AudioInterface::clear() { if(p) p->clear(); }
|
|
||||||
AudioInterface::AudioInterface() : p(nullptr) {}
|
|
||||||
AudioInterface::~AudioInterface() { term(); }
|
AudioInterface::~AudioInterface() { term(); }
|
||||||
|
auto AudioInterface::cap(const string& name) -> bool { return p ? p->cap(name) : false; }
|
||||||
|
auto AudioInterface::get(const string& name) -> any { return p ? p->get(name) : false; }
|
||||||
|
auto AudioInterface::set(const string& name, const any& value) -> bool { return p ? p->set(name, value) : false; }
|
||||||
|
auto AudioInterface::sample(uint16_t left, uint16_t right) -> void { if(p) p->sample(left, right); }
|
||||||
|
auto AudioInterface::clear() -> void { if(p) p->clear(); }
|
||||||
|
|
||||||
/* InputInterface */
|
/* InputInterface */
|
||||||
|
|
||||||
|
|
|
@ -19,47 +19,45 @@ namespace ruby {
|
||||||
#include <ruby/input.hpp>
|
#include <ruby/input.hpp>
|
||||||
|
|
||||||
struct VideoInterface {
|
struct VideoInterface {
|
||||||
void driver(const char* driver = "");
|
|
||||||
const char* optimalDriver();
|
|
||||||
const char* safestDriver();
|
|
||||||
const char* availableDrivers();
|
|
||||||
bool init();
|
|
||||||
void term();
|
|
||||||
|
|
||||||
bool cap(const nall::string& name);
|
|
||||||
nall::any get(const nall::string& name);
|
|
||||||
bool set(const nall::string& name, const nall::any& value);
|
|
||||||
|
|
||||||
bool lock(uint32_t*& data, unsigned& pitch, unsigned width, unsigned height);
|
|
||||||
void unlock();
|
|
||||||
void clear();
|
|
||||||
void refresh();
|
|
||||||
|
|
||||||
VideoInterface();
|
|
||||||
~VideoInterface();
|
~VideoInterface();
|
||||||
|
|
||||||
|
auto driver(nall::string driver = "") -> void;
|
||||||
|
auto optimalDriver() -> nall::string;
|
||||||
|
auto safestDriver() -> nall::string;
|
||||||
|
auto availableDrivers() -> nall::string;
|
||||||
|
auto init() -> bool;
|
||||||
|
auto term() -> void;
|
||||||
|
|
||||||
|
auto cap(const nall::string& name) -> bool;
|
||||||
|
auto get(const nall::string& name) -> nall::any;
|
||||||
|
auto set(const nall::string& name, const nall::any& value) -> bool;
|
||||||
|
|
||||||
|
auto lock(uint32_t*& data, unsigned& pitch, unsigned width, unsigned height) -> bool;
|
||||||
|
auto unlock() -> void;
|
||||||
|
auto clear() -> void;
|
||||||
|
auto refresh() -> void;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Video* p = nullptr;
|
Video* p = nullptr;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct AudioInterface {
|
struct AudioInterface {
|
||||||
void driver(const char* driver = "");
|
|
||||||
const char* optimalDriver();
|
|
||||||
const char* safestDriver();
|
|
||||||
const char* availableDrivers();
|
|
||||||
bool init();
|
|
||||||
void term();
|
|
||||||
|
|
||||||
bool cap(const nall::string& name);
|
|
||||||
nall::any get(const nall::string& name);
|
|
||||||
bool set(const nall::string& name, const nall::any& value);
|
|
||||||
|
|
||||||
void sample(uint16_t left, uint16_t right);
|
|
||||||
void clear();
|
|
||||||
|
|
||||||
AudioInterface();
|
|
||||||
~AudioInterface();
|
~AudioInterface();
|
||||||
|
|
||||||
|
auto driver(nall::string driver = "") -> void;
|
||||||
|
auto optimalDriver() -> nall::string;
|
||||||
|
auto safestDriver() -> nall::string;
|
||||||
|
auto availableDrivers() -> nall::string;
|
||||||
|
auto init() -> bool;
|
||||||
|
auto term() -> void;
|
||||||
|
|
||||||
|
auto cap(const nall::string& name) -> bool;
|
||||||
|
auto get(const nall::string& name) -> nall::any;
|
||||||
|
auto set(const nall::string& name, const nall::any& value) -> bool;
|
||||||
|
|
||||||
|
auto sample(uint16_t left, uint16_t right) -> void;
|
||||||
|
auto clear() -> void;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Audio* p = nullptr;
|
Audio* p = nullptr;
|
||||||
};
|
};
|
||||||
|
@ -83,7 +81,6 @@ struct InputInterface {
|
||||||
auto acquire() -> bool;
|
auto acquire() -> bool;
|
||||||
auto unacquire() -> bool;
|
auto unacquire() -> bool;
|
||||||
auto acquired() -> bool;
|
auto acquired() -> bool;
|
||||||
|
|
||||||
auto poll() -> nall::vector<nall::shared_pointer<nall::HID::Device>>;
|
auto poll() -> nall::vector<nall::shared_pointer<nall::HID::Device>>;
|
||||||
auto rumble(uint64_t id, bool enable) -> bool;
|
auto rumble(uint64_t id, bool enable) -> bool;
|
||||||
|
|
||||||
|
|
|
@ -1,25 +1,24 @@
|
||||||
struct Video {
|
struct Video {
|
||||||
static const char* Handle;
|
static const nall::string Handle;
|
||||||
static const char* Synchronize;
|
static const nall::string Synchronize;
|
||||||
static const char* Depth;
|
static const nall::string Depth;
|
||||||
static const char* Filter;
|
static const nall::string Filter;
|
||||||
static const char* Shader;
|
static const nall::string Shader;
|
||||||
|
|
||||||
static const unsigned FilterNearest;
|
static const unsigned FilterNearest;
|
||||||
static const unsigned FilterLinear;
|
static const unsigned FilterLinear;
|
||||||
|
|
||||||
virtual bool cap(const nall::string& name) { return false; }
|
virtual ~Video() = default;
|
||||||
virtual nall::any get(const nall::string& name) { return false; }
|
|
||||||
virtual bool set(const nall::string& name, const nall::any& value) { return false; }
|
|
||||||
|
|
||||||
virtual bool lock(uint32_t*& data, unsigned& pitch, unsigned width, unsigned height) { return false; }
|
virtual auto cap(const nall::string& name) -> bool { return false; }
|
||||||
virtual void unlock() {}
|
virtual auto get(const nall::string& name) -> nall::any { return false; }
|
||||||
|
virtual auto set(const nall::string& name, const nall::any& value) -> bool { return false; }
|
||||||
|
|
||||||
virtual void clear() {}
|
virtual auto lock(uint32_t*& data, unsigned& pitch, unsigned width, unsigned height) -> bool { return false; }
|
||||||
virtual void refresh() {}
|
virtual auto unlock() -> void {}
|
||||||
virtual bool init() { return true; }
|
virtual auto clear() -> void {}
|
||||||
virtual void term() {}
|
virtual auto refresh() -> void {}
|
||||||
|
|
||||||
Video() {}
|
virtual auto init() -> bool { return true; }
|
||||||
virtual ~Video() {}
|
virtual auto term() -> void {}
|
||||||
};
|
};
|
||||||
|
|
|
@ -24,6 +24,7 @@ ConfigurationManager::ConfigurationManager() {
|
||||||
append(video, "Video");
|
append(video, "Video");
|
||||||
|
|
||||||
audio.append(audio.driver, "Driver");
|
audio.append(audio.driver, "Driver");
|
||||||
|
audio.append(audio.device, "Device");
|
||||||
audio.append(audio.synchronize, "Synchronize");
|
audio.append(audio.synchronize, "Synchronize");
|
||||||
audio.append(audio.mute, "Mute");
|
audio.append(audio.mute, "Mute");
|
||||||
append(audio, "Audio");
|
append(audio, "Audio");
|
||||||
|
|
|
@ -27,6 +27,7 @@ struct ConfigurationManager : Configuration::Document {
|
||||||
|
|
||||||
struct Audio : Configuration::Node {
|
struct Audio : Configuration::Node {
|
||||||
string driver;
|
string driver;
|
||||||
|
string device;
|
||||||
bool synchronize = true;
|
bool synchronize = true;
|
||||||
bool mute = false;
|
bool mute = false;
|
||||||
} audio;
|
} audio;
|
||||||
|
|
|
@ -1,5 +1,16 @@
|
||||||
//request from emulation core to load non-volatile media folder
|
//request from emulation core to load non-volatile media folder
|
||||||
auto Program::loadRequest(unsigned id, string name, string type) -> void {
|
auto Program::loadRequest(unsigned id, string name, string type) -> void {
|
||||||
|
string location = BrowserDialog()
|
||||||
|
.setParent(*presentation)
|
||||||
|
.setTitle({"Load ", name})
|
||||||
|
.setPath({config().library.location, name})
|
||||||
|
.setFilters({string{name, "|*.", type}})
|
||||||
|
.selectFolder();
|
||||||
|
if(!directory::exists(location)) return;
|
||||||
|
|
||||||
|
mediaPaths(id) = location;
|
||||||
|
folderPaths.append(location);
|
||||||
|
emulator->load(id);
|
||||||
}
|
}
|
||||||
|
|
||||||
//request from emulation core to load non-volatile media file
|
//request from emulation core to load non-volatile media file
|
||||||
|
|
|
@ -36,6 +36,7 @@ Program::Program() {
|
||||||
if(!video.init()) { video.driver("None"); video.init(); }
|
if(!video.init()) { video.driver("None"); video.init(); }
|
||||||
|
|
||||||
audio.driver(config().audio.driver);
|
audio.driver(config().audio.driver);
|
||||||
|
audio.set(Audio::Device, config().audio.device);
|
||||||
audio.set(Audio::Handle, presentation->viewport.handle());
|
audio.set(Audio::Handle, presentation->viewport.handle());
|
||||||
audio.set(Audio::Synchronize, config().audio.synchronize);
|
audio.set(Audio::Synchronize, config().audio.synchronize);
|
||||||
audio.set(Audio::Frequency, 96000u);
|
audio.set(Audio::Frequency, 96000u);
|
||||||
|
|
|
@ -8,6 +8,12 @@ HotkeySettings::HotkeySettings(TabFrame* parent) : TabFrameItem(parent) {
|
||||||
mappingList.onChange([&] {
|
mappingList.onChange([&] {
|
||||||
eraseButton.setEnabled((bool)mappingList.selected());
|
eraseButton.setEnabled((bool)mappingList.selected());
|
||||||
});
|
});
|
||||||
|
resetButton.setText("Reset").onActivate([&] {
|
||||||
|
if(MessageDialog("Are you sure you want to erase all hotkey mappings?").setParent(*settingsManager).question() == 0) {
|
||||||
|
for(auto& mapping : inputManager->hotkeys) mapping->unbind();
|
||||||
|
refreshMappings();
|
||||||
|
}
|
||||||
|
});
|
||||||
eraseButton.setText("Erase").onActivate([&] {
|
eraseButton.setText("Erase").onActivate([&] {
|
||||||
if(auto item = mappingList.selected()) {
|
if(auto item = mappingList.selected()) {
|
||||||
inputManager->hotkeys[item->offset()]->unbind();
|
inputManager->hotkeys[item->offset()]->unbind();
|
||||||
|
@ -43,6 +49,7 @@ auto HotkeySettings::assignMapping() -> void {
|
||||||
|
|
||||||
if(auto item = mappingList.selected()) {
|
if(auto item = mappingList.selected()) {
|
||||||
activeMapping = inputManager->hotkeys[item->offset()];
|
activeMapping = inputManager->hotkeys[item->offset()];
|
||||||
|
settingsManager->layout.setEnabled(false);
|
||||||
settingsManager->statusBar.setText({"Press a key or button to map [", activeMapping->name, "] ..."});
|
settingsManager->statusBar.setText({"Press a key or button to map [", activeMapping->name, "] ..."});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -54,6 +61,7 @@ auto HotkeySettings::inputEvent(shared_pointer<HID::Device> device, unsigned gro
|
||||||
if(activeMapping->bind(device, group, input, oldValue, newValue)) {
|
if(activeMapping->bind(device, group, input, oldValue, newValue)) {
|
||||||
activeMapping = nullptr;
|
activeMapping = nullptr;
|
||||||
settingsManager->statusBar.setText("");
|
settingsManager->statusBar.setText("");
|
||||||
|
settingsManager->layout.setEnabled(true);
|
||||||
refreshMappings();
|
refreshMappings();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -102,12 +102,12 @@ auto InputSettings::refreshMappings() -> void {
|
||||||
auto InputSettings::assignMapping() -> void {
|
auto InputSettings::assignMapping() -> void {
|
||||||
inputManager->poll(); //clear any pending events first
|
inputManager->poll(); //clear any pending events first
|
||||||
|
|
||||||
auto mapping = mappingList.selected();
|
if(auto mapping = mappingList.selected()) {
|
||||||
activeMapping = activeDevice().mappings[mapping->offset()];
|
activeMapping = activeDevice().mappings[mapping->offset()];
|
||||||
|
settingsManager->layout.setEnabled(false);
|
||||||
//settingsManager->layout.setEnabled(false);
|
|
||||||
settingsManager->statusBar.setText({"Press a key or button to map [", activeMapping->name, "] ..."});
|
settingsManager->statusBar.setText({"Press a key or button to map [", activeMapping->name, "] ..."});
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
auto InputSettings::assignMouseInput(unsigned id) -> void {
|
auto InputSettings::assignMouseInput(unsigned id) -> void {
|
||||||
if(auto mouse = inputManager->findMouse()) {
|
if(auto mouse = inputManager->findMouse()) {
|
||||||
|
@ -130,7 +130,7 @@ auto InputSettings::inputEvent(shared_pointer<HID::Device> device, unsigned grou
|
||||||
if(activeMapping->bind(device, group, input, oldValue, newValue)) {
|
if(activeMapping->bind(device, group, input, oldValue, newValue)) {
|
||||||
activeMapping = nullptr;
|
activeMapping = nullptr;
|
||||||
settingsManager->statusBar.setText("");
|
settingsManager->statusBar.setText("");
|
||||||
//settingsManager->layout.setEnabled(true); //todo: this isn't enabling child widgets properly (bug in hiro)
|
settingsManager->layout.setEnabled(true);
|
||||||
refreshMappings();
|
refreshMappings();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -42,6 +42,7 @@ struct HotkeySettings : TabFrameItem {
|
||||||
ListView mappingList{&layout, Size{~0, ~0}};
|
ListView mappingList{&layout, Size{~0, ~0}};
|
||||||
HorizontalLayout controlLayout{&layout, Size{~0, 0}};
|
HorizontalLayout controlLayout{&layout, Size{~0, 0}};
|
||||||
Widget spacer{&controlLayout, Size{~0, 0}};
|
Widget spacer{&controlLayout, Size{~0, 0}};
|
||||||
|
Button resetButton{&controlLayout, Size{80, 0}};
|
||||||
Button eraseButton{&controlLayout, Size{80, 0}};
|
Button eraseButton{&controlLayout, Size{80, 0}};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue