RLSAlssm
¶
RLSAlssm(cost_terms, steady_state=True, calc_W=True, calc_xi=True, calc_kappa=True, calc_nu=False, filter_form='cascade', backend=None, steady_state_method='schur')
Recursive Least Square Alssm Class to compute and minimize Alssm Cost Functions.
This class uses either a CostSegment, CompositeCost or NDCompositeCost.
The cost function is (Eq. (20) [Wildhaber2018]):
with \(W_k \in \mathbb{R}^{N \times N}\) the Gram Matrix defined by the ALSMS (\(A\), \(c\), \(\alpha\) and \(w\)), \(\xi_k \in \mathbb{R}^{N}\) the cross correlation of the signal with the ALSSM basis, and \(\kappa_k \in \mathbb{R}\) the signal energy weighted under \(\alpha\) and \(w\). Additionally, \(\nu_k\) is the number of weighted samples in the window.
The quantities \(W_k\), \(\xi_k\), \(\kappa_k\) and \(\nu_k\) can be computed either as a forward or as a backward recursion as defined in Eq. (22-25) [Wildhaber2018].
Info
See also [Wildhaber2018] [Wildhaber2019]. Eq. (19)-(21) in [Baeriswyl2025] define the quantities \(\xi^{(q)}(k,y), q\in\{0,1,2\}\), with the following correspondence: \(\mathrm{vec}(W_k) \triangleq \xi^{(2)}(k,\mathbf{1})\) (with \(\mathbf{1}\) the all ones vector), \(\xi_k \triangleq \xi^{(1)}(k,y)\), and \(\kappa_k \triangleq \xi^{(0)}(k,y)\).
Parameters:
-
cost_terms((CostSegment, CompositeCost, NDCompositeCost)) –Cost function to be minimized recursively. See
CostSegment,CompositeCostorNDCompositeCost. -
steady_state(bool, default:True) –Defines if the ALSSM is steady state (not time-varying, e.g. LTI). If so, \(W_k\) reduces to \(W\). Default: True. This happens in case \(w_k = w\), \(\gamma_k = \gamma\) (see Sec. III-I.2 [Wildhaber2018]). Setting this incorrectly may produce silently wrong results.
-
calc_W(bool, default:True) –If True, compute
W. Required forminimize_v,minimize_x, andeval_errors. Default: True. -
calc_xi(bool, default:True) –If True, compute
xiRequired forminimize_v,minimize_x, andeval_errors. Default: True. -
calc_kappa(bool, default:True) –If True, computes
kappaRequired foreval_errors. Can be set to False when only the minimizer is needed. Default: True. -
calc_nu(bool, default:False) –If True, computes \(\nu_k\), the number of weighted samples in the window. Not yet implemented. Default: False.
-
filter_form(str, default:'cascade') –Controls the internal block structure of the recursive filter.
'cascade': cascade block form (default)'parallel': parallel block form
-
backend(str, default:None) –Selects the computational backend for the state-space recursions. If
None, the globally configured backend is used (seeset_backend).'numpy': pure NumPy implementation (default)'lfilter': transfer-function backend usinglfilter'jit': Numba JIT-compiled backend (requiresnumbapackage)
Notes
Setting steady_state=True (default) assumes the window defined by the cost segments
does not change over time (LTI case), which allows \(W_k\) to be precomputed once
as a constant \(W\). Set steady_state=False for time-varying windows
(e.g. near signal boundaries or with sample-dependent weights), at the cost of
a full \(W_k\) calculation at every sample.
Example
Fit a polynomial of degree 2 to a signal using a left-sided window:
>>> import numpy as np
>>> import lmlib as lm
>>> from lmlib.utils.generator import gen_rect
>>> K = 500
>>> y = gen_rect(K, 200, 100)
>>> alssm = lm.AlssmPoly(poly_degree=2)
>>> seg = lm.Segment(a=-20, b=0, direction=lm.FORWARD, g=10)
>>> rls = lm.RLSAlssm(lm.CostSegment(alssm, seg))
>>> y_hat = rls.fit(y)
To save computation when only the signal estimate is needed, disable \(\kappa\):
Methods:
-
filter–Compute the ALSSM cost parameters \(\xi^{(q)}(k,y)\) for \(q \in \{0,1,2\}\) and \(\nu_k\) for the input signal \(y\).
-
convolve–Convolve (or correlate) a signal with a reference state vector in the ALSSM feature space.
-
minimize_v–Minimizes the cost \(J_k(x)\) subject to a linear constraint on the state vector, returning the reduced-dimensional free parameter \(v\).
-
minimize_x–Minimizes the cost \(J_k(x)\) subject to a linear constraint on the state vector, returning the full N-dimensional state vector \(x\).
-
eval_errors–Evaluates the cost function \(J_k(x)\) at given state vectors
xs. -
fit–Method that chains
filter,minimize_v, and signal reconstruction into a single call.
Attributes:
-
xi2– -
xi1– -
xi0– -
W–ndarray: Gram matrix \(W_k\) (const. \(W\) in steady-state mode). Shape(..., N, N). -
xi–ndarray: Signal cross-correlation \(\xi_k\). Shape(..., N). -
kappa–ndarray: Signal energy \(\kappa_k\). Shape(...,). -
nu–ndarray: Effective number of weighted samples \(\nu_k\) in the window. Not yet implemented.
Source code in lmlib/statespace/rls.py
Methods¶
filter
¶
Compute the ALSSM cost parameters \(\xi^{(q)}(k,y)\) for \(q \in \{0,1,2\}\) and \(\nu_k\) for the input signal \(y\).
For CostSegment and CompositeCost (1-D) each quantity is
computed via the recursive equations (22–25) in [Wildhaber2018] for each cost segment by the
selected backend.
For NDCompositeCost (multi-dimensional) the ND cost separates as a
Kronecker product over the per-dimension sub-costs, see Table III in [Baeriswyl2025].
The computation is therefore chained over dimensions:
_nd_xi_q_recursionprocesses the first axis indim_order, reading directly fromyand producingxi_prevof shape(*Ks, N_0**q).- For each subsequent axis
dim_order[l],_nd_xi_q_asterisk_l_recursionextends the trailing axis ofxi_prevfromNq_prevtoNq_prev * N_l**q, accumulating the Kronecker structure dimension by dimension. - After all
Laxes are processed the trailing axis has size \((N_0 \cdots N_{L-1})^q = N_\text{total}^q\).
The final results are stored in W,
xi, and kappa.
Parameters:
-
y(array_like) –Input signal.
- 1-D / CompositeCost / CostSegment — shape
(K,)or(K, Q).Kis the signal length;Qis the ALSSM output dimension. For scalar ALSSMs (Q = 0internally) a plain 1-D array(K,)is accepted and silently reshaped to(K, 1). - ND / NDCompositeCost — shape
(K_0, K_1, ..., K_{L-1})for scalar ALSSMs (Q = 0), or(K_0, ..., K_{L-1}, Q)for vector-output ALSSMs. There must be exactlyLspatial axes, one per sub-cost, regardless of the processing order given bydim_order.
- 1-D / CompositeCost / CostSegment — shape
-
sample_weights(array_like of shape matching ``y`` without the trailing, default:None) –Qaxis, optional. Per-sample scalar weights \(w_i \in [0, 1]\) applied to each observation before accumulation. Must broadcast to the spatial shape ofy. Default: all ones (uniform weighting). -
dim_order(array_like of int of length ``L``, default:None) –Processing order of the signal axes for
NDCompositeCost.dim_order[0]is processed first by_nd_xi_q_recursion;dim_order[1], dim_order[2], ...are each processed in turn by_nd_xi_q_asterisk_l_recursion.The Kronecker structure of the output state vector follows this order: with
dim_order = [d_0, d_1, ..., d_{L-1}], the trailing axis of the minimiser resultxs(seeminimize_x) is arranged as:\[ x_{d_0} \otimes x_{d_1} \otimes \cdots \otimes x_{d_{L-1}} \]where \(x_{d_l}\) is the per-dimension state vector for axis \(d_l\). Changing
dim_ordertherefore permutes the Kronecker blocks inxs, but does not change the scalar cost \(J_k(x)\).Has no effect for
CostSegmentorCompositeCost(those are inherently 1-D). Default:np.arange(L)(axes in natural order).
Notes
Steady-state mode (steady_state=True, the default):
\(W_k\) is constant and precomputed once via
get_steady_state_W before the other
recursions start. This is valid whenever the window is
time-invariant (\(w_k = w\), \(\gamma_k = \gamma\)), which
holds for all standard Segment definitions without
sample-dependent weights. See Section III-I.2 in [Wildhaber2018].
The calc_W flag is ignored in this mode.
Time-varying mode (steady_state=False):
\(W_k\) is computed sample-by-sample via the \(q = 2\)
recursion, which carries the same computational cost as the
\(q = 1\) recursion. Required near signal boundaries or when
sample_weights varies over time. Only supported for 1-D costs;
multi-dimensional costs require steady_state=True (the ND
\(W\) recursion is not yet implemented).
Returns:
-
None–
Raises:
-
ValueError–If
ydoes not have the expected number of spatial axes or output dimensionQ. Ifsample_weightsdoes not match the spatial shape ofy. -
AssertionError–If
dim_orderdoes not have exactlyLelements. Ifsteady_state=FalseandL > 1(ND time-varying W not supported).
Source code in lmlib/statespace/rls.py
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 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 | |
convolve
¶
Convolve (or correlate) a signal with a reference state vector in the ALSSM feature space.
This is a thin convenience wrapper around filter:
it projects the signal \(y\) into the low-dimensional ALSSM feature space
and then contracts the per-sample cross-correlation state \(\xi_k\) with a
fixed reference state vector \(x_\mathrm{ref}\),
which is equivalent to a sliding-window linear filter whose effective impulse response lives in the span of the ALSSM. Because \(\xi_k\) is obtained by a recursive filter, the cost is \(\mathcal{O}(K \cdot N)\) and independent of the effective filter length (the segment window), unlike a direct sample-domain convolution.
Whether the operation realises a convolution or a correlation depends on
\(x_\mathrm{ref}\) and on the segment direction of the underlying cost
(a time-reversed/inverted model gives the convolution, a forward model
the correlation); see the convolution examples in the 50-convolution
folder.
Internally this is self.filter(y, ...) followed by
numpy.tensordot(self.xi, xref, axes=xref.ndim). For a 1-D xref
(shape (N,)) this reduces to self.xi @ xref; a higher-rank
xref (e.g. shape (channels, N) for multichannel signals)
additionally sums over the leading reference axes, reproducing the
channel-wise accumulation \(\sum_j \xi_{k,j} \, x_{\mathrm{ref},j}\).
Parameters:
-
y(array_like) –Input signal, in the same shape as accepted by
filter. -
xref(array_like) –Reference state vector in the ALSSM feature space. Its shape must match the trailing feature axes of
xi:- 1-D
(N,)for a single-channelCostSegment/CompositeCost, or the Kronecker state(N,)of anNDCompositeCost; (channels, N)for a multichannel signal, in which case the channel axis is summed over.
- 1-D
-
sample_weights(array_like, default:None) –Per-sample weights forwarded to
filter. -
dim_order(array_like, default:None) –Processing order of the signal axes for
NDCompositeCost, forwarded tofilter.
Returns:
-
out(ndarray) –Convolution/correlation output with the spatial shape of \(y\) (the trailing feature axes are contracted away). Shape
(K,)for a 1-D signal and(K_0, ..., K_{L-1})for an ND signal.
Notes
Only \(\xi\) is needed, so if this RLSAlssm was constructed with
calc_W=True and/or calc_kappa=True those flags are turned off
(with a warning) on the first call to avoid computing \(W\) and \(\kappa\).
Construct the RLSAlssm with calc_W=False, calc_kappa=False to
silence the warning.
Examples:
>>> import numpy as np
>>> import lmlib as lm
>>> alssm = lm.AlssmPoly(poly_degree=2)
>>> segment = lm.Segment(a=-10, b=10, direction=lm.BACKWARD, g=200)
>>> cost = lm.CostSegment(alssm, segment)
>>> rls = lm.RLSAlssm(cost, steady_state=True)
>>> y = np.zeros(100); y[40:61] = 1.0 # a rectangular pulse
>>> xref = np.array([1.0, 0.0, 0.0]) # match the local mean (N = 3)
>>> out = rls.convolve(y, xref)
>>> out.shape
(100,)
Source code in lmlib/statespace/rls.py
487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 | |
minimize_v
¶
Minimizes the cost \(J_k(x)\) subject to a linear constraint on the state vector, returning the reduced-dimensional free parameter \(v\).
The state vector \(x\) is constrained to the affine subspace
with known \(H \in \mathbb{R}^{N \times M}\) and \(h \in \mathbb{R}^N\), and unknown \(v \in \mathbb{R}^M\). Substituting into the cost function (Eq. (21) [Wildhaber2018]) and minimizing over \(v\) yields the closed-form solution (Eq. (69) [Wildhaber2018]):
When H=None and h=None (defaults), the constraint reduces to \(x = v\)
(unconstrained minimization), so \(\hat{v}_k = W_k^{-1} \xi_k\).
Parameters:
-
H(array_like of shape (N, M), default:None) –Constraint matrix mapping the free parameter \(v \in \mathbb{R}^M\) to the state space \(\mathbb{R}^N\). If
None, defaults to the identity matrix \(I_N\) (unconstrained, \(M = N\)). -
h(array_like of shape (N,), default:None) –Constraint offset vector \(h \in \mathbb{R}^N\). If
None, defaults to the zero vector (no offset). -
solver((lstsq, solve, inv), default:'lstsq') –Linear-system solver strategy (default:
'lstsq').
Notes
The distinction between minimize_v and minimize_x is:
minimize_vreturns \(v \in \mathbb{R}^M\) (the free parameter, dimension M).minimize_xreturns \(x = Hv + h \in \mathbb{R}^N\) (the full state vector, dimension N).
When H=None, both methods return the same result since \(x = v\).
When steady_state=True, \(W_k = W\) is constant, so \(H^T W H\) is
inverted once. When steady_state=False, samples where \(H^T W_k H\) is
ill-conditioned (singular to machine precision) are left as NaN.
Returns:
-
v(ndarray of shape (..., M)) –Free parameter \(\hat{v}_k\) minimizing \(J_k(Hv + h)\). The leading dimensions match the signal length and any parallel dimensions of the input. Entries are
NaNwhere \(H^T W_k H\) is ill-conditioned (only possible whensteady_state=False).
Raises:
-
AssertionError–If
Hdoes not haveNrows. Ifhdoes not have lengthN. IfH.T @ W @ His not invertible (only whensteady_state=True).
Solver strategies
Three strategies are available via the solver keyword. All are
closed-form (no iterative steps).
'lstsq' (default)
Steady-state — Computes the economy SVD of \(H^\top W H\)
once and stores the resulting pseudoinverse as an attribute
_pinv_HTWH. Subsequent calls with the same H and h
reuse this cached pseudoinverse, reducing the cost to a single
matrix–vector multiply per time step. This is typically 4–22×
faster than calling numpy.linalg.lstsq with all right-hand
sides at once, because that function recomputes the SVD on every
call even when the matrix has not changed.
Non-steady-state — Attempts a batched LU solve
(numpy.linalg.solve) on the full \((K, N, N)\) stack of
matrices. If any matrix is singular, falls back to a fully
vectorised batched SVD pseudoinverse (numpy.linalg.svd over
the batch dimension) — no Python loop required. This gives a
5–70× speedup over the previous per-sample lstsq loop.
Handles rank-deficient \(H^\top W H\) gracefully in both modes by returning the minimum-norm least-squares solution (Moore–Penrose pseudoinverse).
'solve'
Uses numpy.linalg.solve (LU factorisation) on the invertible
subset and falls back to 'lstsq' for rank-deficient blocks.
Slightly faster than 'lstsq' when \(H^\top W H\) is always
full-rank (the common case), but equally robust otherwise. Does
not cache a pseudoinverse.
'inv' (legacy)
Explicit matrix inversion — the original behaviour. Raises an
AssertionError (steady-state) or leaves entries as nan
(non-steady-state) when \(H^\top W H\) is not invertible.
Kept for backward compatibility and benchmarking.
Source code in lmlib/statespace/rls.py
596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 | |
minimize_x
¶
Minimizes the cost \(J_k(x)\) subject to a linear constraint on the state vector, returning the full N-dimensional state vector \(x\).
Internally calls minimize_v to obtain \(\hat{v}_k\), then reconstructs
the state vector via the constraint (Eq. (66) [Wildhaber2018]):
When H=None and h=None (defaults), the constraint reduces to \(x = v\),
so \(\hat{x}_k = W_k^{-1} \xi_k\) (Table V row 1, [Wildhaber2018]).
Parameters:
-
H(array_like of shape (N, M), default:None) –Constraint matrix mapping the free parameter \(v \in \mathbb{R}^M\) to the state space \(\mathbb{R}^N\). If
None, defaults to the identity matrix \(I_N\) (unconstrained, \(M = N\)). -
h(array_like of shape (N,), default:None) –Constraint offset vector \(h \in \mathbb{R}^N\). If
None, defaults to the zero vector (no offset).
Notes
The distinction between minimize_v and minimize_x is:
minimize_vreturns \(v \in \mathbb{R}^M\) (the free parameter, dimension M).minimize_xreturns \(x = Hv + h \in \mathbb{R}^N\) (the full state vector, dimension N).
When H=None, both methods return the same result since \(x = v\).
Returns:
-
x(ndarray of shape (..., N)) –Optimal state vector \(\hat{x}_k = H\hat{v}_k + h\). The leading dimensions match the signal length and any parallel dimensions of the input. Entries are
NaNwhere \(H^T W_k H\) is ill-conditioned (only possible whensteady_state=False).
Raises:
-
AssertionError–Propagated from
minimize_v: IfHdoes not haveNrows. Ifhdoes not have lengthN. IfH.T @ W @ His not invertible (only whensteady_state=True).
Source code in lmlib/statespace/rls.py
eval_errors
¶
Evaluates the cost function \(J_k(x)\) at given state vectors xs.
Using the expanded form of the cost (Eq. (21) [Wildhaber2018]):
this method computes the scalar cost for each provided state vector without performing any minimization. It is useful for comparing the fit quality of different candidate state vectors, e.g. for computing error ratios or log-cost ratios (LCR) as in Section III.F [Wildhaber2018].
Requires W, xi, and kappa to be available, i.e.
filter must have been called with calc_W=True, calc_xi=True,
and calc_kappa=True (all defaults).
Parameters:
-
xs(array_like of shape (..., N)) –State vector(s) at which to evaluate the cost. The last dimension must be the state dimension N. Leading dimensions are broadcast over the signal samples.
Returns:
-
J(ndarray of shape (...)) –Cost \(J_k(x)\) evaluated at each state vector in
xs. Same shape asxswith the last axis removed.
Source code in lmlib/statespace/rls.py
fit
¶
fit(y, output='y_hat', sample_weights=None, dim_order=None, H=None, h=None, eval_alssm_weights=None, solver='lstsq')
Method that chains filter, minimize_v, and signal reconstruction into a single call.
Executes the following steps in order:
filter— computes the recursive filter quantities \(W_k\), \(\xi_k\), \(\kappa_k\) from the input signaly.minimize_v— solves the constrained minimization \(\hat{v}_k = (H^T W_k H)^{-1} H^T (\xi_k - W_k h)\).- Reconstruction — builds \(\hat{x}_k = H\hat{v}_k + h\) and/or the
signal estimate \(\hat{y}_k = C A^j \hat{x}_k\) depending on
output.
Parameters:
-
y(array_like of shape (K, [Q])) –Input signal. See
filterfor shape details. -
output(str or tuple of str, default:'y_hat') –Selects what is returned. One or more of:
'y_hat'(default) — signal estimate \(\hat{y}_k = CA^j\hat{x}_k\) evaluated via the cost term'seval_alssm_output.'x'— full state vector \(\hat{x}_k = H\hat{v}_k + h\), shape(..., N).'v'— free parameter \(\hat{v}_k\), shape(..., M).
Pass a tuple to return multiple outputs, e.g.
output=('x', 'y_hat'). -
sample_weights(array_like of shape (K,), default:None) –Per-sample weights \(w_i \in [0,1]\). Passed to
filter. Default: all ones. -
dim_order(array_like of int, default:None) –Dimension processing order for
NDCompositeCost. Passed tofilter. Default:np.arange(L). -
H(array_like of shape (N, M), default:None) –Constraint matrix. Passed to
minimize_v. Default: identity (unconstrained). -
h(array_like of shape (N,), default:None) –Constraint offset. Passed to
minimize_v. Default: zero vector. -
eval_alssm_weights(array_like, default:None) –Per-ALSSM output weights used when evaluating \(\hat{y}\).
- For
CompositeCost/CostSegment: shape(M,)(or scalar), forwarded asalssm_weightsto the cost term'seval_alssm_output(which passes them toAlssmSum). - For
NDCompositeCost: shape(L, M), forwarded asnd_alssm_weightstoeval_alssm_output, where element[l, m]scales ALSSMmin signal dimensionl.
If
None, all models contribute equally. - For
Returns:
-
result(ndarray or tuple of ndarray) –If
outputis a single string, returns the corresponding array directly. Ifoutputis a tuple, returns a tuple of arrays in the same order.'y_hat': shape(K, [Q])— signal estimate at every time step.'x': shape(..., N)— optimal state vector.'v': shape(..., M)— optimal free parameter.
Raises:
-
AssertionError–If
outputis empty or contains unknown entries. Propagated fromminimize_vifH/hhave wrong shape orH.T @ W @ His not invertible (whensteady_state=True).
Example
Fit a degree-2 polynomial and return the signal estimate:
Return both the state vector and the signal estimate:Source code in lmlib/statespace/rls.py
962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 | |