aboutsummaryrefslogtreecommitdiff
path: root/include
diff options
context:
space:
mode:
Diffstat (limited to 'include')
-rw-r--r--include/guttersynth.h139
-rw-r--r--include/guttersynth.hpp140
2 files changed, 279 insertions, 0 deletions
diff --git a/include/guttersynth.h b/include/guttersynth.h
new file mode 100644
index 0000000..88d874a
--- /dev/null
+++ b/include/guttersynth.h
@@ -0,0 +1,139 @@
+/*
+ * guttersynth.h
+ * Part of libguttersynth
+ *
+ * Copyright Tom Mudd 2019, Richard Knight 2021, 2022
+ *
+ * Ported to C by Richard Knight
+ * from https://github.com/tommmmudd/guttersynthesis by Tom Mudd
+ *
+ */
+
+#ifndef GUTTER_H
+#define GUTTER_H
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#include <stdio.h>
+#include <stdlib.h>
+
+// support for either single or double precision
+#ifndef FLT
+
+#ifdef USE_FLOAT
+#define FLT float
+#else
+#define FLT double
+#endif
+
+#endif
+
+
+// internal/private dcblock state variables
+typedef struct dcblock_session_ {
+ FLT xm1;
+ FLT ym1;
+ FLT r;
+} dcblock_session;
+
+// internal/private state variables
+typedef struct gutter_internal_ {
+ void (*dealloc)(void*);
+ int samplerate;
+ FLT finalY;
+ FLT duffX;
+ FLT duffY;
+ FLT dx;
+ FLT dy;
+ FLT t;
+ FLT* prevX1;
+ FLT* prevX2;
+ FLT* prevY1;
+ FLT* prevY2;
+ FLT* V;
+ FLT* K;
+ FLT* norm;
+ FLT* a0;
+ FLT* a1;
+ FLT* a2;
+ FLT* b1;
+ FLT* b2;
+ FLT* y;
+ dcblock_session dcblock;
+} gutter_internal;
+
+// state with public variables
+typedef struct gutter_state_ {
+ int bankCount;
+ int filterCount;
+
+ FLT gamma;
+ FLT omega;
+ FLT c;
+ FLT dt;
+ FLT singleGain;
+ FLT* gains;
+ FLT* filterFreqs;
+ FLT* Q;
+ int enableAudioInput;
+ int filtersOn;
+ int smoothing;
+ int distortionMethod;
+ gutter_internal gi;
+} gutter_state;
+
+
+// synthesise single sample with input
+FLT gutter_process_input(gutter_state* s, FLT audioInput);
+
+// synthesise single sample
+FLT gutter_process(gutter_state* s);
+
+// synthesise specified number of samples with input
+void gutter_process_input_samples(gutter_state* s, FLT* audioInput, FLT* audioOutput, int nsamps);
+
+// synthesise specified number of samples
+void gutter_process_samples(gutter_state* s, FLT* audioOutput, int nsamps);
+
+// reset state
+void gutter_reset(gutter_state* s);
+
+// clean up memory etc
+void gutter_cleanup(gutter_state* s);
+
+// calculate filter coefficients
+void gutter_calccoeffs(gutter_state* s);
+
+// initialise with custom memory allocator and deallocator
+gutter_state* gutter_init_ca(int bankCount, int filterCount, int samplerate,
+ void* (*allocator)(size_t), void (*deallocator)(void*));
+
+// initialise with standard malloc and free
+gutter_state* gutter_init(int bankCount, int filterCount, int samplerate);
+
+// randomise filters
+void gutter_randomisefilters(gutter_state* s);
+
+// set distortion method
+int gutter_setdistortionmethod(gutter_state* s, int method);
+
+// set frequency of a filter in a bank
+int gutter_setfreq(gutter_state*, int bank, int filter, FLT value);
+
+// set Q of a filter in a bank
+int gutter_setq(gutter_state* s, int bank, int filter, FLT value);
+
+// set gain of a filter in a bank
+int gutter_setgain(gutter_state* s, int bank, int filter, FLT value);
+
+// set Q of all filters
+void gutter_setqall(gutter_state* s, FLT value);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* GUTTER_H */
+
diff --git a/include/guttersynth.hpp b/include/guttersynth.hpp
new file mode 100644
index 0000000..7b961f1
--- /dev/null
+++ b/include/guttersynth.hpp
@@ -0,0 +1,140 @@
+/*
+ * Thin C++ wrapper around the C library
+ */
+#ifndef GUTTERSYNTH_HPP
+#define GUTTERSYNTH_HPP
+
+#include "guttersynth.h"
+#include <stdexcept>
+#include <limits.h>
+
+class GutterSynth {
+protected:
+ gutter_state* s;
+public:
+
+ GutterSynth(int bankCount, int filterCount, int samplerate) {
+ s = gutter_init(bankCount, filterCount, samplerate);
+ }
+
+ GutterSynth(int bankCount, int filterCount, int samplerate,
+ void* (*allocator)(size_t), void (*deallocator)(void*)
+ ) {
+ s = gutter_init_ca(bankCount, filterCount, samplerate,
+ allocator, deallocator);
+ }
+
+ ~GutterSynth() {
+ gutter_cleanup(s);
+ }
+
+ // simple accessors
+ FLT gamma() { return s->gamma; }
+ void gamma(FLT value) { s->gamma = value; }
+
+ FLT omega() { return s->omega; }
+ void omega(FLT value) { s->omega = value; }
+
+ FLT c() { return s->c; }
+ void c(FLT value) { s->c = value; }
+
+ FLT dt() { return s->dt; }
+ void dt(FLT value) { s->dt = value; }
+
+ FLT singleGain() { return s->singleGain; }
+ void singleGain(FLT value) { s->singleGain = value; }
+
+ int filterCount() { return s->filterCount; }
+ int bankCount() { return s->bankCount; }
+
+ bool enableAudioInput() { return bool(s->enableAudioInput); }
+ void enableAudioInput(bool value) { s->enableAudioInput = int(value); }
+
+ bool filtersOn() { return bool(s->filtersOn); }
+ void filtersOn(bool value) { s->filtersOn = int(value); }
+
+ int smoothing() { return s->smoothing; }
+ void smoothing(int smoothing) { s->smoothing = smoothing; }
+
+ int distortionMethod() { return s->distortionMethod; }
+ void distortionMethod(int method) {
+ int result = gutter_setdistortionmethod(s, method);
+ if (result != 0) {
+ throw std::invalid_argument("Filter out of range");
+ }
+ }
+
+ void randomiseFilters() {
+ gutter_randomisefilters(s);
+ }
+
+ void calcCoeffs() {
+ gutter_calccoeffs(s);
+ }
+
+ void setFreq(int bank, int filter, FLT value, bool recalculate=true) {
+ int result = gutter_setfreq(s, bank, filter, value);
+ if (result != 0) {
+ throw std::invalid_argument("Filter out of range");
+ }
+
+ if (recalculate) {
+ calcCoeffs();
+ }
+ }
+
+ void setQ(int bank, int filter, FLT value, bool recalculate=true) {
+ int result = gutter_setq(s, bank, filter, value);
+ if (result != 0) {
+ throw std::invalid_argument("Filter out of range");
+ }
+
+ if (recalculate) {
+ calcCoeffs();
+ }
+ }
+
+ void setQ(FLT value, bool recalculate=true) {
+ gutter_setqall(s, value);
+
+ if (recalculate) {
+ calcCoeffs();
+ }
+ }
+
+ void setGain(int bank, int filter, FLT value, bool recalculate=true) {
+ int result = gutter_setgain(s, bank, filter, value);
+ if (result != 0) {
+ throw std::invalid_argument("Filter out of range");
+ }
+
+ if (recalculate) {
+ calcCoeffs();
+ }
+ }
+
+ void reset() {
+ gutter_reset(s);
+ }
+
+ // process audio members
+ FLT process() {
+ return gutter_process(s);
+ }
+
+ FLT process(FLT audioInput) {
+ return gutter_process_input(s, audioInput);
+ }
+
+ void process(FLT* audioOutput, int nsamps) {
+ gutter_process_samples(s, audioOutput, nsamps);
+ }
+
+ void process(FLT* audioInput, FLT* audioOutput, int nsamps) {
+ gutter_process_input_samples(s, audioInput, audioOutput, nsamps);
+ }
+};
+
+
+#endif /* GUTTERSYNTH_HPP */
+