aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorDaniil Cherednik <dan.cherednik@gmail.com>2019-09-19 00:44:18 +0300
committerDaniil Cherednik <dan.cherednik@gmail.com>2019-09-19 00:44:18 +0300
commit3e33324de42366389678eca16250687b33aa5692 (patch)
tree3a262041f9c47df32b356baa936ebaa1a5cfab35
parent1600f372e233413fa662a3526752a531d7bb7fdd (diff)
downloadlibpqf-3e33324de42366389678eca16250687b33aa5692.tar.gz
Initial implementation of polyphase quadrature filterbank
-rw-r--r--include/libpqf.h52
-rw-r--r--src/pqf.c267
-rw-r--r--tools/plot/data/README.md2
-rw-r--r--tools/plot/data/at3plus.txt385
-rw-r--r--tools/plot/data/mpc.txt513
-rw-r--r--tools/plot/main.py158
-rw-r--r--tools/plot/pqf/pqf_binding.c257
-rw-r--r--tools/plot/pqf/setup.py6
-rw-r--r--tools/plot/run.sh12
9 files changed, 1652 insertions, 0 deletions
diff --git a/include/libpqf.h b/include/libpqf.h
new file mode 100644
index 0000000..a9c05af
--- /dev/null
+++ b/include/libpqf.h
@@ -0,0 +1,52 @@
+/*
+ * This file is part of libpqf project.
+ *
+ * libpqf is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * libpqf 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
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with libpqf; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+ */
+
+#ifndef LIBPQF_H
+#define LIBPQF_H
+
+#include <stdint.h>
+
+typedef struct pqf_a_ctx *pqf_a_ctx_t;
+
+typedef enum {
+ PQF_SUCCESS,
+ PQF_WRONG_SUBBAND_SZ,
+ PQF_WRONG_SUBBANDS_NUM,
+ PQF_WRONG_PROTO_SZ,
+ PQF_FRAME_TOO_LONG,
+ PQF_CONTRACT_VIOLATION,
+ PQF_NOMEM
+} pqf_status_t;
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+pqf_status_t pqf_create_a_ctx(uint16_t subband_sz, uint16_t subbands_num, uint16_t proto_sz, float* proto, pqf_a_ctx_t* ctx);
+void pqf_free_a_ctx(pqf_a_ctx_t ctx);
+
+uint16_t pqf_get_frame_sz(pqf_a_ctx_t ctx);
+uint16_t pqf_get_subband_sz(pqf_a_ctx_t ctx);
+uint16_t pqf_get_subbands_num(pqf_a_ctx_t ctx);
+void pqf_do_analyse(pqf_a_ctx_t ctx, const float* in, float* out);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
diff --git a/src/pqf.c b/src/pqf.c
new file mode 100644
index 0000000..a6e255b
--- /dev/null
+++ b/src/pqf.c
@@ -0,0 +1,267 @@
+/*
+ * This file is part of libpqf.
+ *
+ * libpqf is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * AtracDEnc 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
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with AtracDEnc; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+ */
+
+/*
+ * Musepack pqf implementation was used as reference.
+ * see svn.musepack.net
+ */
+
+#include <stdlib.h>
+
+#include <string.h>
+#include <math.h>
+
+#include "libpqf.h"
+
+struct pqf_a_ctx {
+ float* buf;
+ float* proto;
+ float* y;
+ float* c;
+ float* m;
+
+ uint16_t proto_sz;
+ uint16_t subbands_num;
+ uint16_t subband_size;
+
+ uint16_t proto_step;
+};
+
+static inline uint16_t get_frame_sz(pqf_a_ctx_t ctx)
+{
+ return ctx->subbands_num * ctx->subband_size;
+}
+
+static void init(pqf_a_ctx_t ctx)
+{
+ uint16_t i, j;
+ const float* const t = ctx->proto;
+ uint16_t subbands_num = ctx->subbands_num;
+ float* c = ctx->c;
+ float* m = ctx->m;
+ uint16_t dsb = subbands_num * 2;
+ uint16_t proto_step = ctx->proto_step;
+
+ for (i = 0; i < subbands_num; i++) {
+ for (j = 0; j < subbands_num; j++) {
+ m[i * subbands_num + j] = (float)cos(((2 * i + 1) * j & 127) * M_PI / (subbands_num * 2));
+ }
+ }
+
+ for (i = 0; i < proto_step; i++) {
+ for (j = 0; j < dsb; j++) {
+ if (i & 1) {
+ c[j * proto_step + i] = 1 * t[j + dsb * i];
+ } else {
+ c[j * proto_step + i] = -1 * t[j + dsb * i];
+ }
+ }
+ }
+}
+
+static void vectoring(const float* x, float* y, pqf_a_ctx_t ctx)
+{
+ uint16_t i, j;
+
+ const float* c = ctx->c;
+ uint16_t subbands_num = ctx->subbands_num;
+ uint16_t dsb = ctx->subbands_num << 1;
+ uint16_t proto_step = ctx->proto_step;
+
+ for (i = 0; i < dsb; i++, c += proto_step, x += 1, y += 1 ) {
+ y[0] = 0;
+ for (j = 0; j < proto_step; j++) {
+ y[0] += c[j] * x[j * dsb];
+ }
+ }
+}
+
+static void matrixing(const float* y, float* samples, pqf_a_ctx_t ctx)
+{
+ uint16_t i, j, k;
+ uint16_t subbands_num = ctx->subbands_num;
+ uint16_t subband_size = ctx->subband_size;
+ const float* mi = ctx->m;
+
+ for (i = 0; i < subbands_num; i++, mi += subbands_num, samples += subband_size) {
+ for (j = 0, k = subbands_num / 2; j < subbands_num / 2; j++, k--) {
+ samples[0] += (y[j] + y[subbands_num - j]) * mi[k];
+ }
+ for (j = 1; j < subbands_num / 2; j++) {
+ samples[0] += (y[subbands_num + j] - y[subbands_num * 2 - j]) * mi[subbands_num/2 + j];
+ }
+ samples[0] += y[subbands_num/2];
+ }
+}
+
+static void a_init(pqf_a_ctx_t ctx)
+{
+ init(ctx);
+}
+
+static pqf_status_t check_params(uint16_t subband_sz, uint16_t subbands_num, uint16_t proto_sz)
+{
+ if (!subbands_num || subbands_num > 128) {
+ return PQF_WRONG_SUBBANDS_NUM;
+ }
+
+ if (!proto_sz || proto_sz < subbands_num) {
+ return PQF_WRONG_SUBBAND_SZ;
+ }
+
+ if (!subband_sz || subband_sz > (0x7fff / subbands_num)) {
+ return PQF_FRAME_TOO_LONG;
+ }
+
+ return PQF_SUCCESS;
+}
+
+pqf_status_t pqf_create_a_ctx(uint16_t subband_sz, uint16_t subbands_num, uint16_t proto_sz, float* proto, pqf_a_ctx_t* ctx_result)
+{
+ uint16_t i = 0;
+
+ uint16_t frame_sz = subband_sz * subbands_num;
+ uint16_t extra_sz = proto_sz - subbands_num;
+ uint16_t buf_sz = frame_sz + extra_sz;
+ pqf_status_t status;
+
+ if (ctx_result == NULL)
+ return PQF_CONTRACT_VIOLATION;
+
+ if ((status = check_params(subband_sz, subbands_num, proto_sz)) != PQF_SUCCESS)
+ return status;
+
+ pqf_a_ctx_t ctx = (pqf_a_ctx_t)malloc(sizeof(struct pqf_a_ctx));
+
+ if (!ctx)
+ return PQF_NOMEM;
+
+ ctx->proto_sz = proto_sz;
+ ctx->subbands_num = subbands_num;
+ ctx->subband_size = subband_sz;
+ ctx->proto_step = proto_sz / (subbands_num << 1);
+
+ ctx->buf = (float*)malloc(sizeof(float) * buf_sz);
+
+ if (!ctx->buf)
+ goto nomem_buf;
+
+ for (i = 0; i < buf_sz; i++) {
+ ctx->buf[i] = 0.0;
+ }
+
+ ctx->proto = (float*)malloc(sizeof(float) * proto_sz);
+
+ if (!ctx->proto)
+ goto nomem_proto;
+
+ memcpy(ctx->proto, proto, sizeof(float) * proto_sz);
+
+ ctx->y = (float*)malloc(sizeof(float) * subbands_num * 2);
+
+ if (!ctx->y)
+ goto nomem_y;
+
+ ctx->c = (float*)malloc(sizeof(float) * proto_sz);
+
+ if (!ctx->c)
+ goto nomem_c;
+
+ ctx->m = (float*)malloc(sizeof(float) * subbands_num * subbands_num);
+
+ if (!ctx->c)
+ goto nomem_m;
+
+ a_init(ctx);
+
+ *ctx_result = ctx;
+
+ return PQF_SUCCESS;
+
+nomem_m:
+ free(ctx->c);
+
+nomem_c:
+ free(ctx->y);
+
+nomem_y:
+ free(ctx->proto);
+
+nomem_proto:
+ free(ctx->buf);
+
+nomem_buf:
+ free(ctx);
+
+ return PQF_NOMEM;
+}
+
+uint16_t pqf_get_frame_sz(pqf_a_ctx_t ctx)
+{
+ return get_frame_sz(ctx);
+}
+
+uint16_t pqf_get_subband_sz(pqf_a_ctx_t ctx)
+{
+ return ctx->subband_size;
+}
+
+uint16_t pqf_get_subbands_num(pqf_a_ctx_t ctx)
+{
+ return ctx->subbands_num;
+}
+
+void pqf_free_a_ctx(pqf_a_ctx_t ctx)
+{
+ free(ctx->m);
+ free(ctx->y);
+ free(ctx->c);
+ free(ctx->proto);
+ free(ctx->buf);
+ free(ctx);
+}
+
+void pqf_do_analyse(pqf_a_ctx_t ctx, const float* in, float* out)
+{
+ float* y = ctx->y;
+
+ float* x;
+ const float* pcm;
+ uint16_t n, i;
+
+ float* buf = ctx->buf;
+ uint16_t subband_size = ctx->subband_size;
+ uint16_t subbands_num = ctx->subbands_num;
+ uint16_t frame_sz = subband_size * subbands_num;
+ uint16_t extra_sz = ctx->proto_sz - subbands_num;
+
+ memcpy(buf + frame_sz, buf, extra_sz * sizeof(float));
+ x = buf + frame_sz;
+
+ pcm = in + (subbands_num - 1);
+
+ for (n = 0; n < subband_size; n++, pcm += subbands_num * 2) {
+ x -= subbands_num;
+ for (i = 0; i < subbands_num; i++) {
+ x[i] = *pcm--;
+ }
+ vectoring(x, y, ctx);
+ matrixing (y, &out[n], ctx);
+ }
+}
+
diff --git a/tools/plot/data/README.md b/tools/plot/data/README.md
new file mode 100644
index 0000000..e260a15
--- /dev/null
+++ b/tools/plot/data/README.md
@@ -0,0 +1,2 @@
+Examples of pqf prototype filters
+mpc.txt - Musepack
diff --git a/tools/plot/data/at3plus.txt b/tools/plot/data/at3plus.txt
new file mode 100644
index 0000000..ec72aaf
--- /dev/null
+++ b/tools/plot/data/at3plus.txt
@@ -0,0 +1,385 @@
+#ATRAC3PLUS prototype file
+0.00000058336207,
+0.00000080604229,
+0.00000042005411,
+0.00000004440057,
+-0.00000003226247,
+-0.00000003530856,
+-0.00000001266038,
+-0.00001051678282,
+0.00001183861787,
+-0.00000060053890,
+-0.00000143337536,
+-0.00000231086847,
+-0.00000325697420,
+-0.00000461924219,
+-0.00000638942583,
+-0.00000703029718,
+-0.00000052268200,
+-0.00000091714480,
+-0.00000108977838,
+-0.00000023649704,
+0.00000027792686,
+0.00000052588763,
+0.00000046299544,
+-0.00001084099222,
+-0.00001220357626,
+-0.00002196196510,
+-0.00002134877286,
+-0.00001990710552,
+-0.00001734808211,
+-0.00001198405243,
+-0.00000727012593,
+-0.00000629902070,
+-0.00000916228237,
+-0.00001050293486,
+-0.00000792127867,
+-0.00000417120236,
+-0.00000263366292,
+-0.00000154329177,
+-0.00000057168614,
+0.00000181119538,
+0.00002353085074,
+0.00002780561954,
+0.00003230232323,
+0.00003696891872,
+0.00004157561489,
+0.00004533784522,
+0.00004604394780,
+0.00004858558168,
+0.00000991988691,
+0.00001313118355,
+0.00001478948616,
+0.00001984130540,
+0.00002653474803,
+0.00003386474418,
+0.00004157788862,
+0.00004978773723,
+0.00002968751687,
+0.00003634074528,
+0.00003947730875,
+0.00004292758240,
+0.00004713382441,
+0.00005426778807,
+0.00006174868031,
+0.00006414738891,
+0.00006446454790,
+0.00006830695202,
+0.00007308147178,
+0.00007612784975,
+0.00007485075184,
+0.00007020850899,
+0.00006228515122,
+0.00005827044151,
+0.00005629632869,
+0.00004988881119,
+0.00003561532503,
+0.00001853294270,
+-0.00000166573534,
+-0.00002610587035,
+-0.00005339706695,
+-0.00008079566032,
+-0.00011279431783,
+-0.00014691354590,
+-0.00018116166757,
+-0.00021972040122,
+-0.00025986303808,
+-0.00030001782579,
+-0.00033926189644,
+-0.00036942670704,
+-0.00042499689152,
+-0.00045358928037,
+-0.00048466061708,
+-0.00051135791000,
+-0.00053276919061,
+-0.00054783461383,
+-0.00055472925305,
+-0.00055432377849,
+-0.00054488552269,
+-0.00052537227748,
+-0.00049731286708,
+-0.00045777999912,
+-0.00040612387238,
+-0.00034301576670,
+-0.00026866336702,
+-0.00018248900597,
+-0.00008430792514,
+0.00002508115722,
+0.00014135583478,
+0.00026649952633,
+0.00039945056778,
+0.00053928449051,
+0.00068422866752,
+0.00083093711874,
+0.00097943725996,
+0.00112581404392,
+0.00126854737755,
+0.00140469032340,
+0.00153207418043,
+0.00164810777642,
+0.00175017165020,
+0.00183808000293,
+0.00189900479745,
+0.00194506184198,
+0.00196531717665,
+0.00195937906392,
+0.00192521407735,
+0.00186084536836,
+0.00176518992521,
+0.00163768627681,
+0.00147714314517,
+0.00128322700039,
+0.00105668208562,
+0.00079780723900,
+0.00050782406470,
+0.00018855913368,
+-0.00015771533072,
+-0.00052769453032,
+-0.00091862218687,
+-0.00132635701448,
+-0.00174694834277,
+-0.00217548245564,
+-0.00260676839389,
+-0.00303528923541,
+-0.00345493946224,
+-0.00385913741775,
+-0.00424182694405,
+-0.00459594465792,
+-0.00491465069354,
+-0.00519145652652,
+-0.00541948433965,
+-0.00559201091528,
+-0.00570259056985,
+-0.00574458623305,
+-0.00571426097304,
+-0.00560342334211,
+-0.00540914759040,
+-0.00512683158740,
+-0.00475297635421,
+-0.00428478466347,
+-0.00372032052837,
+-0.00305867753923,
+-0.00229951413348,
+-0.00144354603253,
+-0.00049266568385,
+0.00055068987422,
+0.00168289500289,
+0.00289928726852,
+0.00419431505725,
+0.00556147377938,
+0.00699351215735,
+0.00848235655576,
+0.01001896336675,
+0.01159386243671,
+0.01319687161595,
+0.01481730863452,
+0.01644404232502,
+0.01806553266943,
+0.01967016234994,
+0.02124618366361,
+0.02278177253902,
+0.02426535822451,
+0.02568554319441,
+0.02703126519918,
+0.02829195372760,
+0.02945766039193,
+0.03051890246570,
+0.03146736323833,
+0.03229522705078,
+0.03299576789141,
+0.03356324881315,
+0.03399298712611,
+0.03428143635392,
+0.03442628309131,
+0.03442628309131,
+0.03428143635392,
+0.03399298712611,
+0.03356324881315,
+0.03299576789141,
+0.03229522705078,
+0.03146736323833,
+0.03051890246570,
+0.02945766039193,
+0.02829195372760,
+0.02703126519918,
+0.02568554319441,
+0.02426535822451,
+0.02278177253902,
+0.02124618366361,
+0.01967016234994,
+0.01806553266943,
+0.01644404232502,
+0.01481730863452,
+0.01319687161595,
+0.01159386243671,
+0.01001896336675,
+0.00848235655576,
+0.00699351215735,
+0.00556147377938,
+0.00419431505725,
+0.00289928726852,
+0.00168289500289,
+0.00055068987422,
+-0.00049266568385,
+-0.00144354603253,
+-0.00229951413348,
+-0.00305867753923,
+-0.00372032052837,
+-0.00428478466347,
+-0.00475297635421,
+-0.00512683158740,
+-0.00540914759040,
+-0.00560342334211,
+-0.00571426097304,
+-0.00574458623305,
+-0.00570259056985,
+-0.00559201091528,
+-0.00541948433965,
+-0.00519145652652,
+-0.00491465069354,
+-0.00459594465792,
+-0.00424182694405,
+-0.00385913741775,
+-0.00345493946224,
+-0.00303528923541,
+-0.00260676839389,
+-0.00217548245564,
+-0.00174694834277,
+-0.00132635701448,
+-0.00091862218687,
+-0.00052769453032,
+-0.00015771533072,
+0.00018855913368,
+0.00050782406470,
+0.00079780723900,
+0.00105668208562,
+0.00128322700039,
+0.00147714314517,
+0.00163768627681,
+0.00176518992521,
+0.00186084536836,
+0.00192521407735,
+0.00195937906392,
+0.00196531717665,
+0.00194506184198,
+0.00189900479745,
+0.00183808000293,
+0.00175017165020,
+0.00164810777642,
+0.00153207418043,
+0.00140469032340,
+0.00126854737755,
+0.00112581404392,
+0.00097943725996,
+0.00083093711874,
+0.00068422866752,
+0.00053928449051,
+0.00039945056778,
+0.00026649952633,
+0.00014135583478,
+0.00002508115722,
+-0.00008430792514,
+-0.00018248900597,
+-0.00026866336702,
+-0.00034301576670,
+-0.00040612387238,
+-0.00045777999912,
+-0.00049731286708,
+-0.00052537227748,
+-0.00054488552269,
+-0.00055432377849,
+-0.00055472925305,
+-0.00054783461383,
+-0.00053276919061,
+-0.00051135791000,
+-0.00048466061708,
+-0.00045358928037,
+-0.00042499689152,
+-0.00036942670704,
+-0.00033926189644,
+-0.00030001782579,
+-0.00025986303808,
+-0.00021972040122,
+-0.00018116166757,
+-0.00014691354590,
+-0.00011279431783,
+-0.00008079566032,
+-0.00005339706695,
+-0.00002610587035,
+-0.00000166573534,
+0.00001853294270,
+0.00003561532503,
+0.00004988881119,
+0.00005629632869,
+0.00005827044151,
+0.00006228515122,
+0.00007020850899,
+0.00007485075184,
+0.00007612784975,
+0.00007308147178,
+0.00006830695202,
+0.00006446454790,
+0.00006414738891,
+0.00006174868031,
+0.00005426778807,
+0.00004713382441,
+0.00004292758240,
+0.00003947730875,
+0.00003634074528,
+0.00002968751687,
+0.00004978773723,
+0.00004157788862,
+0.00003386474418,
+0.00002653474803,
+0.00001984130540,
+0.00001478948616,
+0.00001313118355,
+0.00000991988691,
+0.00004858558168,
+0.00004604394780,
+0.00004533784522,
+0.00004157561489,
+0.00003696891872,
+0.00003230232323,
+0.00002780561954,
+0.00002353085074,
+0.00000181119538,
+-0.00000057168614,
+-0.00000154329177,
+-0.00000263366292,
+-0.00000417120236,
+-0.00000792127867,
+-0.00001050293486,
+-0.00000916228237,
+-0.00000629902070,
+-0.00000727012593,
+-0.00001198405243,
+-0.00001734808211,
+-0.00001990710552,
+-0.00002134877286,
+-0.00002196196510,
+-0.00001220357626,
+-0.00001084099222,
+0.00000046299544,
+0.00000052588763,
+0.00000027792686,
+-0.00000023649704,
+-0.00000108977838,
+-0.00000091714480,
+-0.00000052268200,
+-0.00000703029718,
+-0.00000638942583,
+-0.00000461924219,
+-0.00000325697420,
+-0.00000231086847,
+-0.00000143337536,
+-0.00000060053890,
+0.00001183861787,
+-0.00001051678282,
+-0.00000001266038,
+-0.00000003530856,
+-0.00000003226247,
+0.00000004440057,
+0.00000042005411,
+0.00000080604229,
+0.00000058336207
diff --git a/tools/plot/data/mpc.txt b/tools/plot/data/mpc.txt
new file mode 100644
index 0000000..71691ae
--- /dev/null
+++ b/tools/plot/data/mpc.txt
@@ -0,0 +1,513 @@
+# mpc prototype file
+0.00000000000000,
+-0.00000047683716,
+-0.00000047683716,
+-0.00000047683716,
+-0.00000047683716,
+-0.00000047683716,
+-0.00000047683716,
+-0.00000095367432,
+-0.00000095367432,
+-0.00000095367432,
+-0.00000095367432,
+-0.00000143051147,
+-0.00000143051147,
+-0.00000190734863,
+-0.00000190734863,
+-0.00000238418579,
+-0.00000238418579,
+-0.00000286102295,
+-0.00000333786011,
+-0.00000333786011,
+-0.00000381469727,
+-0.00000429153442,
+-0.00000476837158,
+-0.00000524520874,
+-0.00000619888306,
+-0.00000667572021,
+-0.00000762939453,
+-0.00000810623169,
+-0.00000905990601,
+-0.00001001358032,
+-0.00001144409180,
+-0.00001239776611,
+-0.00001382827759,
+-0.00001478195190,
+-0.00001668930054,
+-0.00001811981201,
+-0.00001955032349,
+-0.00002145767212,
+-0.00002336502075,
+-0.00002527236938,
+-0.00002765655518,
+-0.00003004074097,
+-0.00003242492676,
+-0.00003480911255,
+-0.00003767013550,
+-0.00004053115845,
+-0.00004339218140,
+-0.00004625320435,
+-0.00004959106445,
+-0.00005292892456,
+-0.00005578994751,
+-0.00005960464478,
+-0.00006294250488,
+-0.00006628036499,
+-0.00007009506226,
+-0.00007343292236,
+-0.00007677078247,
+-0.00008058547974,
+-0.00008392333984,
+-0.00008726119995,
+-0.00009059906006,
+-0.00009346008301,
+-0.00009632110596,
+-0.00009918212891,
+-0.00010156631470,
+-0.00010395050049,
+-0.00010585784912,
+-0.00010728836060,
+-0.00010824203491,
+-0.00010871887207,
+-0.00010871887207,
+-0.00010824203491,
+-0.00010681152344,
+-0.00010538101196,
+-0.00010251998901,
+-0.00009918212891,
+-0.00009536743164,
+-0.00009012222290,
+-0.00008440017700,
+-0.00007772445679,
+-0.00006961822510,
+-0.00006055831909,
+-0.00005054473877,
+-0.00003957748413,
+-0.00002717971802,
+-0.00001382827759,
+0.00000095367432,
+0.00001716613770,
+0.00003433227539,
+0.00005292892456,
+0.00007295608521,
+0.00009393692017,
+0.00011634826660,
+0.00014019012451,
+0.00016546249390,
+0.00019121170044,
+0.00021886825562,
+0.00024747848511,
+0.00027704238892,
+0.00030755996704,
+0.00033903121948,
+0.00037145614624,
+0.00040435791016,
+0.00043821334839,
+0.00047254562378,
+0.00050735473633,
+0.00054216384888,
+0.00057697296143,
+0.00061178207397,
+0.00064659118652,
+0.00068092346191,
+0.00071430206299,
+0.00074720382690,
+0.00077915191650,
+0.00080966949463,
+0.00083875656128,
+0.00086641311646,
+0.00089168548584,
+0.00091505050659,
+0.00093555450439,
+0.00095415115356,
+0.00096893310547,
+0.00098085403442,
+0.00098943710327,
+0.00099420547485,
+0.00099515914917,
+0.00099182128906,
+0.00098371505737,
+0.00097131729126,
+0.00095367431641,
+0.00093078613281,
+0.00090265274048,
+0.00086879730225,
+0.00082921981812,
+0.00078392028809,
+0.00073194503784,
+0.00067424774170,
+0.00061035156250,
+0.00053930282593,
+0.00046253204346,
+0.00037860870361,
+0.00028848648071,
+0.00019168853760,
+0.00008821487427,
+-0.00002145767212,
+-0.00013732910156,
+-0.00025987625122,
+-0.00038814544678,
+-0.00052213668823,
+-0.00066184997559,
+-0.00080680847168,
+-0.00095653533936,
+-0.00111103057861,
+-0.00126981735229,
+-0.00143241882324,
+-0.00159788131714,
+-0.00176668167114,
+-0.00193738937378,
+-0.00211000442505,
+-0.00228309631348,
+-0.00245714187622,
+-0.00263071060181,
+-0.00280332565308,
+-0.00297403335571,
+-0.00314188003540,
+-0.00330686569214,
+-0.00346708297729,
+-0.00362253189087,
+-0.00377178192139,
+-0.00391435623169,
+-0.00404882431030,
+-0.00417470932007,
+-0.00429058074951,
+-0.00439596176147,
+-0.00448989868164,
+-0.00457048416138,
+-0.00463819503784,
+-0.00469112396240,
+-0.00472831726074,
+-0.00474882125854,
+-0.00475215911865,
+-0.00473737716675,
+-0.00470304489136,
+-0.00464916229248,
+-0.00457382202148,
+-0.00447702407837,
+-0.00435781478882,
+-0.00421524047852,
+-0.00404930114746,
+-0.00385856628418,
+-0.00364303588867,
+-0.00340175628662,
+-0.00313472747803,
+-0.00284147262573,
+-0.00252151489258,
+-0.00217485427856,
+-0.00180053710938,
+-0.00139951705933,
+-0.00097131729126,
+-0.00051593780518,
+-0.00003337860107,
+0.00047588348389,
+0.00101184844971,
+0.00157356262207,
+0.00216150283813,
+0.00277423858643,
+0.00341129302979,
+0.00407218933105,
+0.00475645065308,
+0.00546216964722,
+0.00618934631348,
+0.00693702697754,
+0.00770330429077,
+0.00848722457886,
+0.00928783416748,
+0.01010370254517,
+0.01093339920044,
+0.01177501678467,
+0.01262760162354,
+0.01348924636841,
+0.01435852050781,
+0.01523351669312,
+0.01611280441284,
+0.01699447631836,
+0.01787614822388,
+0.01875686645508,
+0.01963424682617,
+0.02050685882568,
+0.02137231826782,
+0.02222871780396,
+0.02307415008545,
+0.02390718460083,
+0.02472543716431,
+0.02552700042725,
+0.02631092071533,
+0.02707386016846,
+0.02781534194946,
+0.02853298187256,
+0.02922487258911,
+0.02989006042480,
+0.03052663803101,
+0.03113269805908,
+0.03170680999756,
+0.03224802017212,
+0.03275489807129,
+0.03322553634644,
+0.03365993499756,
+0.03405570983887,
+0.03441286087036,
+0.03473043441772,
+0.03500699996948,
+0.03524208068848,
+0.03543519973755,
+0.03558635711670,
+0.03569412231445,
+0.03575897216797,
+0.03578090667725,
+0.03575897216797,
+0.03569412231445,
+0.03558635711670,
+0.03543519973755,
+0.03524208068848,
+0.03500699996948,
+0.03473043441772,
+0.03441286087036,
+0.03405570983887,
+0.03365993499756,
+0.03322553634644,
+0.03275489807129,
+0.03224802017212,
+0.03170680999756,
+0.03113269805908,
+0.03052663803101,
+0.02989006042480,
+0.02922487258911,
+0.02853298187256,
+0.02781534194946,
+0.02707386016846,
+0.02631092071533,
+0.02552700042725,
+0.02472543716431,
+0.02390718460083,
+0.02307415008545,
+0.02222871780396,
+0.02137231826782,
+0.02050685882568,
+0.01963424682617,
+0.01875686645508,
+0.01787614822388,
+0.01699447631836,
+0.01611280441284,
+0.01523351669312,
+0.01435852050781,
+0.01348924636841,
+0.01262760162354,
+0.01177501678467,
+0.01093339920044,
+0.01010370254517,
+0.00928783416748,
+0.00848722457886,
+0.00770330429077,
+0.00693702697754,
+0.00618934631348,
+0.00546216964722,
+0.00475645065308,
+0.00407218933105,
+0.00341129302979,
+0.00277423858643,
+0.00216150283813,
+0.00157356262207,
+0.00101184844971,
+0.00047588348389,
+-0.00003337860107,
+-0.00051593780518,
+-0.00097131729126,
+-0.00139951705933,
+-0.00180053710938,
+-0.00217485427856,
+-0.00252151489258,
+-0.00284147262573,
+-0.00313472747803,
+-0.00340175628662,
+-0.00364303588867,
+-0.00385856628418,
+-0.00404930114746,
+-0.00421524047852,
+-0.00435781478882,
+-0.00447702407837,
+-0.00457382202148,
+-0.00464916229248,
+-0.00470304489136,
+-0.00473737716675,
+-0.00475215911865,
+-0.00474882125854,
+-0.00472831726074,
+-0.00469112396240,
+-0.00463819503784,
+-0.00457048416138,
+-0.00448989868164,
+-0.00439596176147,
+-0.00429058074951,
+-0.00417470932007,
+-0.00404882431030,
+-0.00391435623169,
+-0.00377178192139,
+-0.00362253189087,
+-0.00346708297729,
+-0.00330686569214,
+-0.00314188003540,
+-0.00297403335571,
+-0.00280332565308,
+-0.00263071060181,
+-0.00245714187622,
+-0.00228309631348,
+-0.00211000442505,
+-0.00193738937378,
+-0.00176668167114,
+-0.00159788131714,
+-0.00143241882324,
+-0.00126981735229,
+-0.00111103057861,
+-0.00095653533936,
+-0.00080680847168,
+-0.00066184997559,
+-0.00052213668823,
+-0.00038814544678,
+-0.00025987625122,
+-0.00013732910156,
+-0.00002145767212,
+0.00008821487427,
+0.00019168853760,
+0.00028848648071,
+0.00037860870361,
+0.00046253204346,
+0.00053930282593,
+0.00061035156250,
+0.00067424774170,
+0.00073194503784,
+0.00078392028809,
+0.00082921981812,
+0.00086879730225,
+0.00090265274048,
+0.00093078613281,
+0.00095367431641,
+0.00097131729126,
+0.00098371505737,
+0.00099182128906,
+0.00099515914917,
+0.00099420547485,
+0.00098943710327,
+0.00098085403442,
+0.00096893310547,
+0.00095415115356,
+0.00093555450439,
+0.00091505050659,
+0.00089168548584,
+0.00086641311646,
+0.00083875656128,
+0.00080966949463,
+0.00077915191650,
+0.00074720382690,
+0.00071430206299,
+0.00068092346191,
+0.00064659118652,
+0.00061178207397,
+0.00057697296143,
+0.00054216384888,
+0.00050735473633,
+0.00047254562378,
+0.00043821334839,
+0.00040435791016,
+0.00037145614624,
+0.00033903121948,
+0.00030755996704,
+0.00027704238892,
+0.00024747848511,
+0.00021886825562,
+0.00019121170044,
+0.00016546249390,
+0.00014019012451,
+0.00011634826660,
+0.00009393692017,
+0.00007295608521,
+0.00005292892456,
+0.00003433227539,
+0.00001716613770,
+0.00000095367432,
+-0.00001382827759,
+-0.00002717971802,
+-0.00003957748413,
+-0.00005054473877,
+-0.00006055831909,
+-0.00006961822510,
+-0.00007772445679,
+-0.00008440017700,
+-0.00009012222290,
+-0.00009536743164,
+-0.00009918212891,
+-0.00010251998901,
+-0.00010538101196,
+-0.00010681152344,
+-0.00010824203491,
+-0.00010871887207,
+-0.00010871887207,
+-0.00010824203491,
+-0.00010728836060,
+-0.00010585784912,
+-0.00010395050049,
+-0.00010156631470,
+-0.00009918212891,
+-0.00009632110596,
+-0.00009346008301,
+-0.00009059906006,
+-0.00008726119995,
+-0.00008392333984,
+-0.00008058547974,
+-0.00007677078247,
+-0.00007343292236,
+-0.00007009506226,
+-0.00006628036499,
+-0.00006294250488,
+-0.00005960464478,
+-0.00005578994751,
+-0.00005292892456,
+-0.00004959106445,
+-0.00004625320435,
+-0.00004339218140,
+-0.00004053115845,
+-0.00003767013550,
+-0.00003480911255,
+-0.00003242492676,
+-0.00003004074097,
+-0.00002765655518,
+-0.00002527236938,
+-0.00002336502075,
+-0.00002145767212,
+-0.00001955032349,
+-0.00001811981201,
+-0.00001668930054,
+-0.00001478195190,
+-0.00001382827759,
+-0.00001239776611,
+-0.00001144409180,
+-0.00001001358032,
+-0.00000905990601,
+-0.00000810623169,
+-0.00000762939453,
+-0.00000667572021,
+-0.00000619888306,
+-0.00000524520874,
+-0.00000476837158,
+-0.00000429153442,
+-0.00000381469727,
+-0.00000333786011,
+-0.00000333786011,
+-0.00000286102295,
+-0.00000238418579,
+-0.00000238418579,
+-0.00000190734863,
+-0.00000190734863,
+-0.00000143051147,
+-0.00000143051147,
+-0.00000095367432,
+-0.00000095367432,
+-0.00000095367432,
+-0.00000095367432,
+-0.00000047683716,
+-0.00000047683716,
+-0.00000047683716,
+-0.00000047683716,
+-0.00000047683716,
+-0.00000047683716
diff --git a/tools/plot/main.py b/tools/plot/main.py
new file mode 100644
index 0000000..ddf0bce
--- /dev/null
+++ b/tools/plot/main.py
@@ -0,0 +1,158 @@
+#!/usr/bin/env python3
+# -*- coding: utf-8 -*-
+
+import argparse
+import sys
+import os
+import re
+
+import numpy as np
+import matplotlib.pyplot as plt
+import scipy.fftpack
+
+import pqf
+
+RUNS = 2
+SAMPLERATE = 44100
+
+def show_proto(proto):
+
+ fig = plt.figure()
+
+ yf = scipy.fftpack.fft(proto)
+
+ max_freq = SAMPLERATE // 2;
+ f_len = len(proto) // 2
+ freq_step = max_freq / f_len;
+
+ freq = np.empty(f_len)
+
+ for i in range(f_len):
+ freq[i] = i * freq_step
+
+ fig.add_subplot(111).plot(freq, 20 * np.log10(np.abs(yf[:f_len])))
+ fig.add_subplot(222).plot(proto)
+
+ plt.show()
+
+def calc_energy_db(data):
+ res = 0.0
+ for x in data:
+ res += x * x
+
+ return 10 * np.log10(res / len(data))
+
+def do_an(ctx, time, data, shift):
+ data_slice = data[shift : ctx.frame_sz() + shift]
+ time_slice = time[shift : ctx.frame_sz() + shift]
+
+ energy_in = calc_energy_db(data_slice)
+
+ res = ctx.do(data_slice)
+
+ subbands_num = ctx.subbands_num();
+ subband_sz = ctx.subband_sz();
+
+ energy_out = calc_energy_db(res)
+
+ rel = energy_out - energy_in
+
+ attenuations = []
+ for i in range(subbands_num):
+ start = i * subband_sz
+ end = start + subband_sz
+ tmp = calc_energy_db(res[start:end]) - energy_in
+ for j in range(subband_sz):
+ attenuations.append(tmp)
+
+ textstr = 'out - in : %.3f' % rel
+
+ fig = plt.figure()
+
+ color = 'tab:blue'
+ a = fig.add_subplot(111)
+ a.set_ylabel('value', color=color)
+ a.plot(res, color=color)
+ a.set_title("output")
+ props = dict(boxstyle='round', facecolor='wheat', alpha=0.5)
+ a.text(0.01, 0.95, textstr, transform=a.transAxes, fontsize=10,
+ verticalalignment='top', bbox=props)
+
+ ax2 = a.twinx()
+ color = 'tab:red'
+ ax2.set_ylabel('attenuation, db', color=color)
+ ax2.plot(attenuations, color=color)
+
+ b = fig.add_subplot(222)
+
+ textstr = 'in: %.3f' % energy_in
+ b.plot(time_slice, data_slice)
+ b.text(0.5, 0.95, textstr, transform=a.transAxes, fontsize=10,
+ verticalalignment='top', bbox=props)
+
+ b.set_title("input");
+
+ plt.show()
+
+def process_freq(freq, subband_sz, subbands_num, prototype):
+ ctx = pqf.AnalyzeCtx(subband_sz, subbands_num, prototype)
+ frame_sz = ctx.frame_sz()
+ time = np.arange(0, frame_sz * RUNS, 1)
+ amplitude = 1 * np.sin(2 * np.pi * freq * time / SAMPLERATE).astype(np.float32)
+
+ do_an(ctx, time, amplitude, 0)
+ do_an(ctx, time, amplitude, frame_sz)
+
+def show_response(pos, subband_sz, subbands_num, prototype):
+ ctx = pqf.AnalyzeCtx(subband_sz, subbands_num, prototype)
+
+ time = np.arange(0, ctx.frame_sz() * RUNS, 1)
+ amplitude = np.zeros(ctx.frame_sz() * RUNS)
+ amplitude[ctx.frame_sz() + pos] = 1.0
+
+ do_an(ctx, time, amplitude, 0)
+ do_an(ctx, time, amplitude, ctx.frame_sz())
+
+def load_prototype(path):
+ prototype = []
+ with open(path) as fp:
+ for line in fp:
+ if line[0] == "#":
+ continue
+ for val in re.findall(r"[-+]?\d*\.\d+|\d+",line):
+ prototype.append(np.float32(val))
+
+ return prototype
+
+def main():
+ parser = argparse.ArgumentParser()
+
+ modes = ['freq', 'delta', 'proto']
+ parser.add_argument('--mode', choices=modes, required=True, help='run mode')
+ parser.add_argument('--file', type=str, help='path to prototype file', required=True, action='store')
+ parser.add_argument('--subband-size', type=int, help='size of subband', action='store')
+ parser.add_argument('--subbands-num', type=int, help='number of subbands to split', action='store')
+
+ args, free_args = parser.parse_known_args()
+
+ if not os.path.isfile(args.file):
+ print("Prototype file {} does not exist.".format(args.file), file=sys.stderr)
+ sys.exit()
+
+ prototype = load_prototype(args.file)
+
+ if (args.mode == 'proto'):
+ show_proto(prototype)
+ if (args.mode == 'delta'):
+ if len(free_args) != 1:
+ print('one free argument (position) must be provided in delta mode')
+ sys.exit()
+ show_response(int(free_args[0]), args.subband_size, args.subbands_num, prototype)
+ else:
+ if len(free_args) != 1:
+ print('one free argument (frequency) must be provided in frequency mode')
+ sys.exit()
+ process_freq(int(free_args[0]), args.subband_size, args.subbands_num, prototype)
+
+if __name__ == "__main__":
+ main()
diff --git a/tools/plot/pqf/pqf_binding.c b/tools/plot/pqf/pqf_binding.c
new file mode 100644
index 0000000..353e2de
--- /dev/null
+++ b/tools/plot/pqf/pqf_binding.c
@@ -0,0 +1,257 @@
+#include <Python.h>
+#include <structmember.h>
+
+#include <libpqf.h>
+
+typedef struct {
+ PyObject_HEAD;
+ pqf_a_ctx_t ctx;
+} AnalyzeCtxObject;
+
+static PyObject *
+AnalyzeCtx_new(PyTypeObject *type, PyObject *args, PyObject *kwds)
+{
+ PyObject *seq;
+ AnalyzeCtxObject *self;
+ uint32_t i;
+ float* proto;
+ uint32_t proto_sz;
+ uint32_t subband_sz;
+ uint32_t subbands_num;
+
+ self = (AnalyzeCtxObject*)type->tp_alloc(type, 0);
+
+ if (!self) {
+ return PyErr_NoMemory( );
+ }
+
+ if (!PyArg_ParseTuple(args, "iiO",
+ &subband_sz, &subbands_num, &seq))
+ {
+ return NULL;
+ }
+
+ seq = PySequence_Fast(seq, "argument must be iterable");
+
+ if (!seq) {
+ return NULL;
+ }
+
+ proto_sz = PySequence_Fast_GET_SIZE(seq);
+
+ proto = (float*)malloc(proto_sz * sizeof(float));
+
+ if(!proto) {
+ Py_DECREF(seq);
+ return PyErr_NoMemory( );
+ }
+
+ for (i = 0; i < proto_sz; i++) {
+ PyObject *fitem;
+ PyObject *item = PySequence_Fast_GET_ITEM(seq, i);
+ if(!item) {
+ Py_DECREF(seq);
+ free(proto);
+ return NULL;
+ }
+
+ fitem = PyNumber_Float(item);
+ if(!fitem) {
+ Py_DECREF(seq);
+ free(proto);
+ PyErr_SetString(PyExc_TypeError, "all items must be numbers");
+ return NULL;
+ }
+
+ proto[i] = PyFloat_AS_DOUBLE(fitem);
+ Py_DECREF(fitem);
+ }
+
+ pqf_status_t status = pqf_create_a_ctx(subband_sz, subbands_num, proto_sz,
+ proto, &self->ctx);
+
+ free(proto);
+
+ if (status != PQF_SUCCESS) {
+ switch (status) {
+ case PQF_NOMEM:
+ PyErr_SetString(PyExc_MemoryError, "No memory to create analyze context");
+ break;
+ case PQF_WRONG_SUBBAND_SZ:
+ PyErr_SetString(PyExc_ValueError, "Unsupported subband size");
+ break;
+ case PQF_WRONG_SUBBANDS_NUM:
+ PyErr_SetString(PyExc_ValueError, "Unsupported subbands number");
+ break;
+ case PQF_WRONG_PROTO_SZ:
+ PyErr_SetString(PyExc_ValueError, "Unsupported prototype size");
+ break;
+ case PQF_FRAME_TOO_LONG:
+ PyErr_SetString(PyExc_ValueError, "Frame too long");
+ break;
+ case PQF_CONTRACT_VIOLATION:
+ PyErr_SetString(PyExc_ValueError, "Conract violation");
+ break;
+ }
+ return NULL;
+ }
+
+ return (PyObject*)self;
+}
+
+static void
+AnalyzeCtx_dealloc(AnalyzeCtxObject *self)
+{
+ pqf_free_a_ctx(self->ctx);
+}
+
+static PyObject* analyzectx_framesz(PyObject* self, PyObject* args)
+{
+ AnalyzeCtxObject* ctx = (AnalyzeCtxObject*)self;
+ return PyLong_FromUnsignedLong(pqf_get_frame_sz(ctx->ctx));
+}
+
+static PyObject* analyzectx_subbandsz(PyObject* self, PyObject* args)
+{
+ AnalyzeCtxObject* ctx = (AnalyzeCtxObject*)self;
+ return PyLong_FromUnsignedLong(pqf_get_subband_sz(ctx->ctx));
+}
+
+static PyObject* analyzectx_subbandsnum(PyObject* self, PyObject* args)
+{
+ AnalyzeCtxObject* ctx = (AnalyzeCtxObject*)self;
+ return PyLong_FromUnsignedLong(pqf_get_subbands_num(ctx->ctx));
+}
+
+static PyObject* analyzectx_do(PyObject* self, PyObject* args)
+{
+ float* in;
+ float* out;
+
+ uint32_t i = 0;
+ uint32_t len = 0;
+ AnalyzeCtxObject* ctx = (AnalyzeCtxObject*)self;
+ uint32_t frame_sz = pqf_get_frame_sz(ctx->ctx);
+
+ PyObject *seq;
+
+ if (!PyArg_ParseTuple(args, "O", &seq)) {
+ return NULL;
+ }
+
+ seq = PySequence_Fast(seq, "argument must be iterable");
+
+ if (!seq) {
+ return NULL;
+ }
+
+ len = PySequence_Fast_GET_SIZE(seq);
+
+ if (len != frame_sz) {
+ PyErr_SetString(PyExc_ValueError, "wrong given frame size");
+ return NULL;
+ }
+
+ in = (float*)malloc(len * sizeof(float));
+ if(!in) {
+ Py_DECREF(seq);
+ return PyErr_NoMemory( );
+ }
+
+ for (i = 0; i < len; i++) {
+ PyObject *fitem;
+ PyObject *item = PySequence_Fast_GET_ITEM(seq, i);
+ if(!item) {
+ Py_DECREF(seq);
+ free(in);
+ return NULL;
+ }
+ fitem = PyNumber_Float(item);
+ if(!fitem) {
+ Py_DECREF(seq);
+ free(in);
+ PyErr_SetString(PyExc_TypeError, "all items must be numbers");
+ return NULL;
+ }
+ in[i] = PyFloat_AS_DOUBLE(fitem);
+ Py_DECREF(fitem);
+ }
+
+ PyObject *my_list = PyList_New(0);
+ if(my_list == NULL) {
+ Py_DECREF(seq);
+ free(in);
+ return NULL;
+ }
+
+ out = (float*)calloc(pqf_get_frame_sz(ctx->ctx), sizeof(float));
+ if (out == NULL) {
+ Py_DECREF(seq);
+ free(in);
+ return NULL;
+ }
+
+
+ pqf_do_analyse(ctx->ctx, in, out);
+
+ for (i = 0; i < len; i++) {
+ PyObject *o = Py_BuildValue("f", out[i]);
+ if(PyList_Append(my_list, o) == -1) {
+ Py_DECREF(seq);
+ free(in);
+ free(out);
+ return NULL;
+ }
+ }
+
+ free(in);
+ free(out);
+ return my_list;
+}
+
+static PyMethodDef methods[] = {
+ { NULL, NULL, 0, NULL }
+};
+
+static PyMethodDef AnalyzeCtx_methods[] = {
+ {"do", (PyCFunction)analyzectx_do, METH_VARARGS, NULL},
+ {"frame_sz", (PyCFunction)analyzectx_framesz, METH_NOARGS, NULL},
+ {"subband_sz", (PyCFunction)analyzectx_subbandsz, METH_NOARGS, NULL},
+ {"subbands_num", (PyCFunction)analyzectx_subbandsnum, METH_NOARGS, NULL},
+ {NULL}
+};
+
+static PyTypeObject AnalyzeCtxType = {
+ PyVarObject_HEAD_INIT(NULL, 0)
+ .tp_name = "pqf.AnalyzeCtx",
+ .tp_doc = "AnalyzeCtx",
+ .tp_basicsize = sizeof(AnalyzeCtxObject),
+ .tp_itemsize = 0,
+ .tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE,
+ .tp_new = AnalyzeCtx_new,
+ .tp_dealloc = (destructor)AnalyzeCtx_dealloc,
+ .tp_methods = AnalyzeCtx_methods,
+};
+
+static struct PyModuleDef module = {
+ PyModuleDef_HEAD_INIT,
+ .m_name = "pqf",
+ .m_doc = "pqf library test binding",
+ .m_size = -1,
+ .m_methods = methods
+};
+
+PyMODINIT_FUNC PyInit_pqf(void)
+{
+ PyObject *m;
+ if (PyType_Ready(&AnalyzeCtxType) < 0)
+ return NULL;
+
+ m = PyModule_Create(&module);
+ if (m == NULL)
+ return NULL;
+
+ Py_INCREF(&AnalyzeCtxType);
+ PyModule_AddObject(m, "AnalyzeCtx", (PyObject*)&AnalyzeCtxType);
+ return m;
+}
diff --git a/tools/plot/pqf/setup.py b/tools/plot/pqf/setup.py
new file mode 100644
index 0000000..8ccd042
--- /dev/null
+++ b/tools/plot/pqf/setup.py
@@ -0,0 +1,6 @@
+from distutils.core import setup, Extension
+
+setup(name = 'pqf', version = '1.0', \
+ ext_modules = [Extension('pqf', \
+ sources=['pqf_binding.c', '../../../src/pqf.c'], \
+ include_dirs=['../../../include/'])])
diff --git a/tools/plot/run.sh b/tools/plot/run.sh
new file mode 100644
index 0000000..06e7788
--- /dev/null
+++ b/tools/plot/run.sh
@@ -0,0 +1,12 @@
+#!/usr/bin/env bash
+
+set -e
+
+cd pqf
+
+python3.7 setup.py build
+
+cd ..
+
+export PYTHONPATH=./pqf/build/lib.mingw-3.7/
+./main.py "$@"