forked from RhysU/ar
-
Notifications
You must be signed in to change notification settings - Fork 0
/
ar-python.cpp
451 lines (398 loc) · 17.1 KB
/
ar-python.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
// Copyright (C) 2012, 2013 Rhys Ulerich
//
// This Source Code Form is subject to the terms of the Mozilla Public
// License, v. 2.0. If a copy of the MPL was not distributed with this
// file, You can obtain one at http://mozilla.org/MPL/2.0/.
#include <Python.h>
#include <numpy/arrayobject.h>
#include <cstdlib>
#include <string>
#include <vector>
#include "ar.hpp"
// Permit Python 2 or 3 compilation
#if PY_MAJOR_VERSION < 3
# define PyUnicode_FromString PyString_FromString
# define PyInt_FromSize_t PyLong_FromSize_t
#endif
/** @file
* A Python extension module for exposing autoregressive model utilities.
*/
// FIXME Add window_T0-like option per arsel.cpp
// Compile-time defaults in the code also appearing in arsel docstring
#define DEFAULT_SUBMEAN true
#define DEFAULT_ABSRHO true
#define DEFAULT_CRITERION "CIC"
#define DEFAULT_MINORDER 0
#define DEFAULT_MAXORDER 512
#define STRINGIFY(x) STRINGIFY_HELPER(x)
#define STRINGIFY_HELPER(x) #x
// TODO The returned values are no longer keys but attributes
static const char ar_arsel_docstring[] =
" Usage: M = arsel (data, submean, absrho, criterion, minorder, maxorder)\n"
"\n"
" Automatically fit autoregressive models to input signals.\n"
"\n"
" Use ar::burg_method and ar::best_model to fit an autoregressive process\n"
" for signals contained in the rows of matrix data. Sample means will\n"
" be subtracted whenever submean is true. Model orders minorder through\n"
" min(columns(data), maxorder) will be considered. A dictionary is\n"
" returned where each key either contains a result indexable by the\n"
" signal number (i.e. the row indices of input matrix data) or it contains\n"
" a single scalar applicable to all signals.\n"
"\n"
" The model order will be selected using the specified criterion.\n"
" Criteria are specified using the following abbreviations:\n"
" AIC - Akaike information criterion\n"
" AICC - asymptotically-corrected Akaike information criterion\n"
" BIC - consistent criterion BIC\n"
" CIC - combined information criterion\n"
" FIC - finite information criterion\n"
" FSIC - finite sample information criterion\n"
" GIC - generalized information criterion\n"
" MCC - minimally consistent criterion\n"
"\n"
" The number of samples in data (i.e. the number of rows) is returned\n"
" in key 'N'. The filter()-ready process parameters are returned\n"
" in key 'AR', the sample mean in 'mu', and the innovation variance\n"
" \\sigma^2_\\epsilon in 'sigma2eps'. The process output variance\n"
" \\sigma^2_\\x and process gain are returned in keys 'sigma2x' and\n"
" 'gain', respectively. Autocorrelations for lags zero through the\n"
" model order, inclusive, are returned in key 'autocor'. The raw\n"
" signals are made available for later use in field 'data'.\n"
"\n"
" Given the observed autocorrelation structure, a decorrelation time\n"
" 'T0' is computed by ar::decorrelation_time and used to estimate\n"
" the effective signal variance 'eff_var'. The number of effectively\n"
" independent samples is returned in 'eff_N'. These effective values are\n"
" combined to estimate the sampling error (i.e. the standard deviation\n"
" of the sample mean) as field 'mu_sigma'. The absolute value of the\n"
" autocorrelation function will be used in computing the decorrelation\n"
" times whenever absrho is true.\n"
"\n"
" For example, given a sequence or *row-vector* of samples 'd', one can fit\n"
" a process, obtain its poles, and simulate a realization of length M using\n"
"\n"
" from ar import arsel\n"
" from numpy import reciprocal, roots\n"
" from numpy.random import randn\n"
" from scipy.signal import lfilter\n"
" a = arsel(d)\n"
" p = [reciprocal(roots(model[::-1])) for model in a.AR]\n"
" x = a.mu[0] + lfilter([1], a.AR[0], sqrt(a.sigma2eps[0])*randn(M))\n"
"\n"
" Continuing the example, one can compute the model autocorrelation\n"
" function at lags 0:M using\n"
"\n"
" from numpy import array, ones\n"
" from scipy.signal import lfiltic\n"
" zi = lfiltic([1], a.AR[0], a.autocor[0])\n"
" rho = ones(M+1)\n"
" rho[1:] = lfilter([1], a.AR[0], zeros(M), zi=zi)[0]\n"
"\n"
" When omitted, submean defaults to " STRINGIFY(DEFAULT_SUBMEAN) ".\n"
" When omitted, absrho defaults to " STRINGIFY(DEFAULT_ABSRHO) ".\n"
" When omitted, criterion defaults to " STRINGIFY(DEFAULT_CRITERION) ".\n"
" When omitted, minorder defaults to " STRINGIFY(DEFAULT_MINORDER) ".\n"
" When omitted, maxorder defaults to " STRINGIFY(DEFAULT_MAXORDER) ".\n"
;
// Return type for ar_arsel initialized within initar
static PyTypeObject *ar_ArselType = NULL;
extern "C" PyObject *ar_arsel(PyObject *self, PyObject *args)
{
PyObject *ret_args = NULL, *ret = NULL;
// Sanity check that initar could build ar_ArselType
if (!PyType_Check(ar_ArselType)) {
PyErr_SetString(PyExc_RuntimeError,
"PyType_Check(ar_ArselType) failed");
return NULL;
}
// Prepare argument storage with default values
PyObject *data_obj = NULL;
int submean = DEFAULT_SUBMEAN;
int absrho = DEFAULT_ABSRHO;
const char *criterion = DEFAULT_CRITERION;
std::size_t minorder = DEFAULT_MINORDER;
std::size_t maxorder = DEFAULT_MAXORDER;
// Parse input tuple with second and subsequent arguments optional
{
unsigned long ul_minorder = minorder, ul_maxorder = maxorder;
if (!PyArg_ParseTuple(args, "O|iiskk", &data_obj, &submean, &absrho,
&criterion,
&ul_minorder, &ul_maxorder)) {
return NULL;
}
minorder = ul_minorder;
maxorder = ul_maxorder;
}
// Lookup the desired model selection criterion
typedef ar::best_model_function<
ar::Burg,npy_intp,npy_intp,std::vector<double>
> best_model_function;
const best_model_function::type best_model
= best_model_function::lookup(std::string(criterion), submean);
if (!best_model) {
PyErr_SetString(PyExc_RuntimeError,
"Unknown model selection criterion provided to arsel.");
return NULL;
}
// Incoming data may be noncontiguous but should otherwise be well-behaved
// On success, 'data' is returned so Py_DECREF is applied only on failure
PyObject *data = PyArray_FROMANY(data_obj, NPY_DOUBLE, 1, 2,
NPY_ALIGNED | NPY_ELEMENTSTRIDES | NPY_NOTSWAPPED);
if (!data) {
Py_DECREF(data);
return NULL;
}
// Reshape any 1D ndarray into a 2D ndarray organized as a row vector.
// Permits the remainder of the routine to uniformly worry about 2D only.
if (1 == ((PyArrayObject *)(data))->nd) {
npy_intp dim[2] = { 1, PyArray_DIM(data, 0) };
PyArray_Dims newshape = { dim, sizeof(dim)/sizeof(dim[0]) };
PyObject *olddata = data;
data = PyArray_Newshape((PyArrayObject *)olddata,
&newshape, NPY_ANYORDER);
Py_DECREF(olddata);
if (!data) {
Py_DECREF(data);
PyErr_SetString(PyExc_RuntimeError,
"Unable to reorganize data into matrix-like row vector.");
return NULL;
}
}
// How many data points are there?
npy_intp M = PyArray_DIM(data, 0);
npy_intp N = PyArray_DIM(data, 1);
// Prepare per-signal storage locations to return to caller
// TODO Ensure these invocations all worked as expected
PyObject *_AR = PyList_New(M);
PyObject *_autocor = PyList_New(M);
PyObject *_eff_N = PyArray_ZEROS(1, &M, NPY_DOUBLE, 0);
PyObject *_eff_var = PyArray_ZEROS(1, &M, NPY_DOUBLE, 0);
PyObject *_gain = PyArray_ZEROS(1, &M, NPY_DOUBLE, 0);
PyObject *_mu = PyArray_ZEROS(1, &M, NPY_DOUBLE, 0);
PyObject *_mu_sigma = PyArray_ZEROS(1, &M, NPY_DOUBLE, 0);
PyObject *_sigma2eps = PyArray_ZEROS(1, &M, NPY_DOUBLE, 0);
PyObject *_sigma2x = PyArray_ZEROS(1, &M, NPY_DOUBLE, 0);
PyObject *_T0 = PyArray_ZEROS(1, &M, NPY_DOUBLE, 0);
// Prepare vectors to capture burg_method() output
std::vector<double> params, sigma2e, gain, autocor;
params .reserve(maxorder*(maxorder + 1)/2);
sigma2e.reserve(maxorder + 1);
gain .reserve(maxorder + 1);
autocor.reserve(maxorder + 1);
// Prepare repeatedly-used working storage for burg_method()
std::vector<double> f, b, Ak, ac;
// Process each signal in turn...
for (npy_intp i = 0; i < M; ++i)
{
// Use burg_method to estimate a hierarchy of AR models from input data
params .clear();
sigma2e.clear();
gain .clear();
autocor.clear();
ar::strided_adaptor<const double*> signal_begin(
(const double*) PyArray_GETPTR2(data, i, 0),
PyArray_STRIDES(data)[1] / sizeof(double));
ar::strided_adaptor<const double*> signal_end (
(const double*) PyArray_GETPTR2(data, i, N),
PyArray_STRIDES(data)[1] / sizeof(double));
ar::burg_method(signal_begin, signal_end,
*(double*)PyArray_GETPTR1(_mu, i),
maxorder,
std::back_inserter(params),
std::back_inserter(sigma2e),
std::back_inserter(gain),
std::back_inserter(autocor),
submean, /* output hierarchy? */ true, f, b, Ak, ac);
// Keep only best model per chosen criterion via function pointer
try {
best_model(N, minorder, params, sigma2e, gain, autocor);
}
catch (std::exception &e)
{
PyErr_SetString(PyExc_RuntimeError, e.what());
goto fail;
}
// Compute decorrelation time from the estimated autocorrelation model
ar::predictor<double> p = ar::autocorrelation(
params.begin(), params.end(), gain[0], autocor.begin());
*(double*)PyArray_GETPTR1(_T0, i)
= ar::decorrelation_time(N, p, absrho);
// Filter()-ready process parameters in field 'AR' with leading one
PyObject *_ARi = PyList_New(params.size() + 1);
PyList_SET_ITEM(_AR, i, _ARi);
PyList_SET_ITEM(_ARi, 0, PyFloat_FromDouble(1));
for (std::size_t k = 0; k < params.size(); ++k) {
PyList_SET_ITEM(_ARi, k+1, PyFloat_FromDouble(params[k]));
}
// Field 'sigma2eps'
*(double*)PyArray_GETPTR1(_sigma2eps, i) = sigma2e[0];
// Field 'gain'
*(double*)PyArray_GETPTR1(_gain, i) = gain[0];
// Field 'sigma2x'
*(double*)PyArray_GETPTR1(_sigma2x, i) = gain[0]*sigma2e[0];
// Field 'autocor'
PyObject *_autocori = PyList_New(autocor.size());
PyList_SET_ITEM(_autocor, i, _autocori);
for (std::size_t k = 0; k < autocor.size(); ++k) {
PyList_SET_ITEM(_autocori, k, PyFloat_FromDouble(autocor[k]));
}
// Field 'eff_var'
// Unbiased effective variance expression from [Trenberth1984]
*(double*)PyArray_GETPTR1(_eff_var, i)
= (N*gain[0]*sigma2e[0]) / (N - *(double*)PyArray_GETPTR1(_T0, i));
// Field 'eff_N'
*(double*)PyArray_GETPTR1(_eff_N, i)
= N / *(double*)PyArray_GETPTR1(_T0, i);
// Field 'mu_sigma'
// Variance of the sample mean using effective quantities
*(double*)PyArray_GETPTR1(_mu_sigma, i)
= std::sqrt( *(double*)PyArray_GETPTR1(_eff_var, i)
/ *(double*)PyArray_GETPTR1(_eff_N, i));
}
// Prepare build and return an ar_ArselType via tuple constructor
// See initar(...) method for the collections.namedtuple-based definition
ret_args = PyTuple_Pack(17, PyBool_FromLong(absrho),
_AR,
_autocor,
PyUnicode_FromString(criterion),
data,
_eff_N,
_eff_var,
_gain,
PyLong_FromSize_t(maxorder),
PyLong_FromSize_t(minorder),
_mu,
_mu_sigma,
PyLong_FromLong(N),
_sigma2eps,
_sigma2x,
PyBool_FromLong(submean),
_T0);
if (!ret_args) {
PyErr_SetString(PyExc_RuntimeError,
"Unable to prepare arguments used to build return value.");
goto fail;
}
ret = PyObject_CallObject((PyObject *)ar_ArselType, ret_args);
if (!ret) {
PyErr_SetString(PyExc_RuntimeError,
"Unable to construct return value.");
goto fail;
}
return ret;
fail:
Py_XDECREF(ret_args);
Py_XDECREF(ret);
Py_XDECREF(data);
Py_XDECREF(_AR);
Py_XDECREF(_autocor);
Py_XDECREF(_eff_N);
Py_XDECREF(_eff_var);
Py_XDECREF(_gain);
Py_XDECREF(_mu);
Py_XDECREF(_mu_sigma);
Py_XDECREF(_sigma2eps);
Py_XDECREF(_sigma2x);
Py_XDECREF(_T0);
return NULL;
}
// Specification of methods available in the module
static PyMethodDef ar_methods[] = {
{"arsel", ar_arsel, METH_VARARGS, ar_arsel_docstring},
{NULL, NULL, 0, NULL}
};
// Module docstring
static const char ar_docstring[] = "Autoregressive process modeling tools";
// Use collections.namedtuple() to make type 'Arsel' for ar_arsel use
// The tuple is ordered alphabetically in case-insensitive manner
static void init_namedtuple(void)
{
PyObject *modName = PyUnicode_FromString((char *)"collections");
PyObject *mod = PyImport_Import(modName);
PyObject *func = PyObject_GetAttrString(mod,(char*)"namedtuple");
PyObject *args = PyTuple_Pack(2,
PyUnicode_FromString((char *) "Arsel"),
PyUnicode_FromString((char *) " absrho"
" AR"
" autocor"
" criterion"
" data"
" eff_N"
" eff_var"
" gain"
" maxorder"
" minorder"
" mu"
" mu_sigma"
" N"
" sigma2eps"
" sigma2x"
" submean"
" T0"));
ar_ArselType = (PyTypeObject *) PyObject_CallObject(func, args);
Py_DECREF(args);
Py_DECREF(func);
Py_DECREF(mod);
Py_DECREF(modName);
}
// -----------------------------------------------------------------------------
// For both Python 2 and Python 3 compatibility, the remainder of file follows
// https://docs.python.org/3/howto/cporting.html#module-initialization-and-state
// -----------------------------------------------------------------------------
struct module_state {
PyObject *error;
};
#if PY_MAJOR_VERSION >= 3
#define GETSTATE(m) ((struct module_state*)PyModule_GetState(m))
#else
#define GETSTATE(m) (&_state)
static struct module_state _state;
#endif
#if PY_MAJOR_VERSION >= 3
static int ar_traverse(PyObject *m, visitproc visit, void *arg) {
Py_VISIT(GETSTATE(m)->error);
return 0;
}
static int ar_clear(PyObject *m) {
Py_CLEAR(GETSTATE(m)->error);
return 0;
}
static struct PyModuleDef moduledef = {
PyModuleDef_HEAD_INIT,
"ar",
ar_docstring,
sizeof(struct module_state),
ar_methods,
NULL,
ar_traverse,
ar_clear,
NULL
};
#define INITERROR return NULL
extern "C" PyMODINIT_FUNC PyInit_ar(void)
#else
#define INITERROR return
extern "C" void initar(void)
#endif
{
#if PY_MAJOR_VERSION >= 3
PyObject *module = PyModule_Create(&moduledef);
#else
PyObject *module = Py_InitModule3("ar", ar_methods, ar_docstring);
#endif
if (module == NULL)
INITERROR;
struct module_state *st = GETSTATE(module);
st->error = PyErr_NewException("ar.Error", NULL, NULL);
if (st->error == NULL) {
Py_DECREF(module);
INITERROR;
}
// Register NumPy and the Arsel namedtuple
import_array();
init_namedtuple();
#if PY_MAJOR_VERSION >= 3
return module;
#endif
}