001/*******************************************************************************
002 * This software is provided as a supplement to the authors' textbooks on digital
003 * image processing published by Springer-Verlag in various languages and editions.
004 * Permission to use and distribute this software is granted under the BSD 2-Clause
005 * "Simplified" License (see http://opensource.org/licenses/BSD-2-Clause).
006 * Copyright (c) 2006-2023 Wilhelm Burger, Mark J. Burge. All rights reserved.
007 * Visit https://imagingbook.com for additional details.
008 ******************************************************************************/
009
010package imagingbook.spectral.dct;
011
012/**
013 * <p>
014 * Calculates the 1D DFT using tabulated cosine values for {@code float} or
015 * {@code double} data (see sub-classes {@link Dct1dDirect.Float} and
016 * {@link Dct1dDirect.Double}, respectively). This implementation is
017 * considerably faster than the naive version without cosine tables (see
018 * {@link Dct1dSlow}). Other optimizations are possible. Note that this class
019 * has no public constructor - instantiate sub-class {@link Dct1dDirect.Float}
020 * or {@link Dct1dDirect.Double} instead, as shown below. See Sec. 20.1 of [1]
021 * for additional details.
022 * </p>
023 * <p>
024 * Usage example (for {@code float} data):
025 * </p>
026 * <pre>
027 * float[] data = {1, 2, 3, 4, 5, 3, 0};
028 * Dct1d.Float dct = new Dct1dDirect.Float(data.length);
029 * dct.forward(data);
030 * dct.inverse(data);
031 * ...
032 * </pre>
033 * <p>
034 * [1] W. Burger, M.J. Burge, <em>Digital Image Processing &ndash; An Algorithmic
035 * Introduction</em>, 3rd ed, Springer (2022).
036 * </p>
037 * 
038 * @author WB
039 * @version 2022/10/23
040 * @see Dct1dSlow
041 * @see Dct1dFast
042 */
043public abstract class Dct1dDirect extends Dct1dImp {
044
045        final double CM0 = 1.0 / Math.sqrt(2);
046        final int M4;                           // = 4*M
047        final double s;                         // common scale factor
048        final double[] cosTable;        // pre-calculated table of cosines
049
050        private Dct1dDirect(int M) {
051                super(M);
052                this.M4 = 4 * M;
053                this.s = Math.sqrt(2.0 / M); 
054                this.cosTable = makeCosineTable();
055        }
056
057        private double[] makeCosineTable() {
058                double[] table = new double[M4];        // we need a table of size 4*M
059                for (int j = 0; j < M4; j++) {          // j is equivalent to (m * (2 * u + 1)) % 4M
060                        //double phi = j * Math.PI / (2 * M);
061                        double phi = 2 * j * Math.PI / M4;
062                        table[j] = Math.cos(phi);
063                }
064                return table;
065        }
066        
067        // ------------------------------------------------------------------------------
068        
069        /**
070         * One-dimensional DCT implementation using {@code float} data. 
071         */
072        public static class Float extends Dct1dDirect implements Dct1d.Float {
073                
074                private final float[] tmp;              // array to hold temporary data
075                
076                /**
077                 * Constructor.
078                 * @param M the data size
079                 */
080                public Float(int M) {
081                        super(M);
082                        this.tmp = new float[M];
083                }
084
085                @Override
086                public void forward(float[] g) {
087                        checkSize(g);
088                        if (g.length != M)
089                                throw new IllegalArgumentException();
090                        float[] G = tmp;
091                        for (int m = 0; m < M; m++) {
092                                double cm = (m == 0) ? CM0 : 1.0;
093                                double sum = 0;
094                                for (int u = 0; u < M; u++) {
095                                        sum += g[u] * cm * cosTable[(m * (2 * u + 1)) % M4];
096                                }
097                                G[m] = (float) (s * sum);
098                        }
099                        System.arraycopy(G, 0, g, 0, M); // copy G -> g
100                }
101
102                @Override
103                public void inverse(float[] G) {
104                        checkSize(G);
105                        if (G.length != M)
106                                throw new IllegalArgumentException();
107                        float[] g = tmp;
108                        for (int u = 0; u < M; u++) {
109                                double sum = 0;
110                                for (int m = 0; m < M; m++) {
111                                        double cm = (m == 0) ? CM0 : 1.0;
112                                        sum += cm * G[m] * cosTable[(m * (2 * u + 1)) % M4];
113                                }
114                                g[u] = (float) (s * sum);
115                        }
116                        System.arraycopy(g, 0, G, 0, M); // copy g -> G
117                } 
118                
119        }
120
121        // ------------------------------------------------------------------------------
122        
123        /**
124         * One-dimensional DCT implementation using {@code double} data. 
125         */
126        public static class Double extends Dct1dDirect implements Dct1d.Double {
127                
128                private final double[] tmp;             // array to hold temporary data
129                
130                /**
131                 * Constructor.
132                 * @param M the data size
133                 */
134                public Double(int M) {
135                        super(M);
136                        this.tmp = new double[M];
137                }
138
139                @Override
140                public void forward(double[] g) {
141                        checkSize(g);
142                        if (g.length != M)
143                                throw new IllegalArgumentException();
144                        double[] G = tmp;
145                        for (int m = 0; m < M; m++) {
146                                double cm = (m == 0) ? CM0 : 1.0;
147                                double sum = 0;
148                                for (int u = 0; u < M; u++) {
149                                        sum += g[u] * cm * cosTable[(m * (2 * u + 1)) % M4];
150                                }
151                                G[m] = s * sum;
152                        }
153                        System.arraycopy(G, 0, g, 0, M); // copy G -> g
154                }
155
156                @Override
157                public void inverse(double[] G) {
158                        checkSize(G);
159                        if (G.length != M)
160                                throw new IllegalArgumentException();
161                        double[] g = tmp;
162                        for (int u = 0; u < M; u++) {
163                                double sum = 0;
164                                for (int m = 0; m < M; m++) {
165                                        double cm = (m == 0) ? CM0 : 1.0;
166                                        sum += cm * G[m] * cosTable[(m * (2 * u + 1)) % M4];
167                                }
168                                g[u] = s * sum;
169                        }
170                        System.arraycopy(g, 0, G, 0, M); // copy g -> G
171                }
172        }
173
174}