Add encode/decode opus bindings
This commit is contained in:
parent
9f870836dc
commit
bc1f4f1a49
|
@ -0,0 +1,6 @@
|
||||||
|
# Godot 4+ specific ignores
|
||||||
|
.godot/
|
||||||
|
export
|
||||||
|
addons/**/~*.dll
|
||||||
|
.sconsign.dblite
|
||||||
|
*.obj
|
|
@ -0,0 +1,45 @@
|
||||||
|
#!/usr/bin/env python
|
||||||
|
from glob import glob
|
||||||
|
from pathlib import Path
|
||||||
|
from SCons.Script import MSVSProject
|
||||||
|
|
||||||
|
env = SConscript("godot-cpp/SConstruct")
|
||||||
|
|
||||||
|
# Sources
|
||||||
|
env.Append(CPPPATH=["src/"])
|
||||||
|
sources = Glob("src/*.cpp")
|
||||||
|
|
||||||
|
# Opus (Windows x64)
|
||||||
|
env.Append(CPPPATH=['#3rdparty/opus/include'])
|
||||||
|
env.Append(LIBPATH=['#3rdparty/opus/lib'])
|
||||||
|
env.Append(LIBS=['opus'])
|
||||||
|
|
||||||
|
(extension_path,) = glob("export/addons/*/*.gdextension")
|
||||||
|
addon_path = Path(extension_path).parent
|
||||||
|
project_name = Path(extension_path).stem
|
||||||
|
debug_or_release = "release" if env["target"] == "template_release" else "debug"
|
||||||
|
|
||||||
|
if env["platform"] == "macos":
|
||||||
|
library = env.SharedLibrary(
|
||||||
|
"{0}/lib/lib{1}.{2}.{3}.framework/{1}.{2}.{3}".format(
|
||||||
|
addon_path,
|
||||||
|
project_name,
|
||||||
|
env["platform"],
|
||||||
|
debug_or_release,
|
||||||
|
),
|
||||||
|
source=sources,
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
library = env.SharedLibrary(
|
||||||
|
"{}/lib/lib{}.{}.{}.{}{}".format(
|
||||||
|
addon_path,
|
||||||
|
project_name,
|
||||||
|
env["platform"],
|
||||||
|
debug_or_release,
|
||||||
|
env["arch"],
|
||||||
|
env["SHLIBSUFFIX"],
|
||||||
|
),
|
||||||
|
source=sources,
|
||||||
|
)
|
||||||
|
|
||||||
|
Default(library)
|
|
@ -0,0 +1,100 @@
|
||||||
|
#include "GodotOpus.h"
|
||||||
|
|
||||||
|
#include <godot_cpp/core/class_db.hpp>
|
||||||
|
#include <godot_cpp/variant/utility_functions.hpp>
|
||||||
|
|
||||||
|
namespace godot {
|
||||||
|
|
||||||
|
Opus *Opus::singleton = nullptr;
|
||||||
|
constexpr auto sampleFrames = 480;
|
||||||
|
|
||||||
|
void Opus::_bind_methods()
|
||||||
|
{
|
||||||
|
ClassDB::bind_method(D_METHOD("encode"), &Opus::encode);
|
||||||
|
ClassDB::bind_method(D_METHOD("decode"), &Opus::decode);
|
||||||
|
}
|
||||||
|
|
||||||
|
Opus *Opus::get_singleton()
|
||||||
|
{
|
||||||
|
return singleton;
|
||||||
|
}
|
||||||
|
|
||||||
|
Opus::Opus()
|
||||||
|
{
|
||||||
|
ERR_FAIL_COND(singleton != nullptr);
|
||||||
|
singleton = this;
|
||||||
|
|
||||||
|
int err{};
|
||||||
|
|
||||||
|
m_encoder = opus_encoder_create(48000, 1, OPUS_APPLICATION_VOIP, &err);
|
||||||
|
ERR_FAIL_COND(err != OPUS_OK);
|
||||||
|
ERR_FAIL_COND(m_encoder == nullptr);
|
||||||
|
|
||||||
|
m_decoder = opus_decoder_create(48000, 1, &err);
|
||||||
|
ERR_FAIL_COND(err != OPUS_OK);
|
||||||
|
ERR_FAIL_COND(m_decoder == nullptr);
|
||||||
|
|
||||||
|
err = opus_encoder_ctl(m_encoder, OPUS_SET_BANDWIDTH(OPUS_BANDWIDTH_SUPERWIDEBAND));
|
||||||
|
ERR_FAIL_COND(err < 0);
|
||||||
|
|
||||||
|
err = opus_encoder_ctl(m_encoder, OPUS_SET_BITRATE(24000));
|
||||||
|
ERR_FAIL_COND(err < 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
Opus::~Opus()
|
||||||
|
{
|
||||||
|
ERR_FAIL_COND(singleton != this);
|
||||||
|
opus_encoder_destroy(m_encoder);
|
||||||
|
opus_decoder_destroy(m_decoder);
|
||||||
|
singleton = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
PackedFloat32Array Opus::encode(PackedVector2Array input)
|
||||||
|
{
|
||||||
|
if (input.size() < sampleFrames) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<float> data(sampleFrames);
|
||||||
|
for (size_t i = 0; i < sampleFrames; i++) {
|
||||||
|
data[i] = input[i].x;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<unsigned char> output(sampleFrames * 2);
|
||||||
|
const auto r = opus_encode_float(m_encoder, data.data(), sampleFrames, output.data(), output.size());
|
||||||
|
if (r == -1) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
auto outputArray = PackedFloat32Array{};
|
||||||
|
outputArray.resize(r);
|
||||||
|
for (size_t i = 0; i < r; i++) {
|
||||||
|
outputArray[i] = output[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
return outputArray;
|
||||||
|
}
|
||||||
|
|
||||||
|
PackedVector2Array Opus::decode(PackedFloat32Array input)
|
||||||
|
{
|
||||||
|
std::vector<unsigned char> inputData(sampleFrames*2);
|
||||||
|
for (size_t i = 0; i < input.size(); i++) {
|
||||||
|
inputData[i] = input[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<float> output(sampleFrames*2);
|
||||||
|
const auto r = opus_decode_float(m_decoder, inputData.data(), input.size(), output.data(), sampleFrames, 0);
|
||||||
|
if (r != sampleFrames) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
auto packedOutput = PackedVector2Array{};
|
||||||
|
packedOutput.resize(r);
|
||||||
|
for (size_t i = 0; i < r; i++) {
|
||||||
|
packedOutput[i] = Vector2{output[i], output[i]};
|
||||||
|
}
|
||||||
|
|
||||||
|
return packedOutput;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,35 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <godot_cpp/classes/object.hpp>
|
||||||
|
#include <godot_cpp/core/class_db.hpp>
|
||||||
|
#include <godot_cpp/variant/typed_array.hpp>
|
||||||
|
|
||||||
|
#include "opus.h"
|
||||||
|
|
||||||
|
using namespace godot;
|
||||||
|
|
||||||
|
namespace godot {
|
||||||
|
|
||||||
|
class Opus : public Object
|
||||||
|
{
|
||||||
|
GDCLASS(Opus, Object);
|
||||||
|
static Opus *singleton;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
static void _bind_methods();
|
||||||
|
|
||||||
|
public:
|
||||||
|
static Opus *get_singleton();
|
||||||
|
|
||||||
|
Opus();
|
||||||
|
~Opus();
|
||||||
|
|
||||||
|
PackedFloat32Array encode(PackedVector2Array input);
|
||||||
|
PackedVector2Array decode(PackedFloat32Array input);
|
||||||
|
private:
|
||||||
|
OpusEncoder* m_encoder;
|
||||||
|
OpusDecoder* m_decoder;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,52 @@
|
||||||
|
#include "register_types.h"
|
||||||
|
|
||||||
|
#include <gdextension_interface.h>
|
||||||
|
#include <godot_cpp/core/class_db.hpp>
|
||||||
|
#include <godot_cpp/core/defs.hpp>
|
||||||
|
#include <godot_cpp/classes/engine.hpp>
|
||||||
|
#include <godot_cpp/godot.hpp>
|
||||||
|
|
||||||
|
#include <godot_cpp/classes/node.hpp>
|
||||||
|
#include <godot_cpp/core/class_db.hpp>
|
||||||
|
#include <godot_cpp/core/class_db.hpp>
|
||||||
|
#include <godot_cpp/variant/utility_functions.hpp>
|
||||||
|
|
||||||
|
#include "GodotOpus.h"
|
||||||
|
|
||||||
|
using namespace godot;
|
||||||
|
|
||||||
|
static Opus* _godot_opus_singleton;
|
||||||
|
|
||||||
|
void gdextension_initialize(ModuleInitializationLevel p_level)
|
||||||
|
{
|
||||||
|
if (p_level == MODULE_INITIALIZATION_LEVEL_SCENE)
|
||||||
|
{
|
||||||
|
ClassDB::register_class<Opus>();
|
||||||
|
|
||||||
|
_godot_opus_singleton = memnew(Opus);
|
||||||
|
Engine::get_singleton()->register_singleton("Opus", Opus::get_singleton());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void gdextension_terminate(ModuleInitializationLevel p_level)
|
||||||
|
{
|
||||||
|
if (p_level == MODULE_INITIALIZATION_LEVEL_SCENE)
|
||||||
|
{
|
||||||
|
Engine::get_singleton()->unregister_singleton("Opus");
|
||||||
|
memdelete(_godot_opus_singleton);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
extern "C"
|
||||||
|
{
|
||||||
|
GDExtensionBool GDE_EXPORT gdextension_init(GDExtensionInterfaceGetProcAddress p_get_proc_address, GDExtensionClassLibraryPtr p_library, GDExtensionInitialization *r_initialization)
|
||||||
|
{
|
||||||
|
godot::GDExtensionBinding::InitObject init_obj(p_get_proc_address, p_library, r_initialization);
|
||||||
|
|
||||||
|
init_obj.register_initializer(gdextension_initialize);
|
||||||
|
init_obj.register_terminator(gdextension_terminate);
|
||||||
|
init_obj.set_minimum_library_initialization_level(MODULE_INITIALIZATION_LEVEL_SCENE);
|
||||||
|
|
||||||
|
return init_obj.init();
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,8 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <godot_cpp/core/class_db.hpp>
|
||||||
|
|
||||||
|
using namespace godot;
|
||||||
|
|
||||||
|
void gdextension_initialize(ModuleInitializationLevel p_level);
|
||||||
|
void gdextension_terminate(ModuleInitializationLevel p_level);
|
Loading…
Reference in New Issue