github.com/gopherd/gonum@v0.0.4/dsp/fourier/radix24.go (about)

     1  // Copyright ©2020 The Gonum Authors. All rights reserved.
     2  // Use of this source code is governed by a BSD-style
     3  // license that can be found in the LICENSE file.
     4  
     5  package fourier
     6  
     7  import (
     8  	"math"
     9  	"math/bits"
    10  	"math/cmplx"
    11  )
    12  
    13  // CoefficientsRadix2 computes the Fourier coefficients of the input
    14  // sequence, converting the time series in seq into the frequency spectrum,
    15  // in place and returning it. This transform is unnormalized; a call to
    16  // CoefficientsRadix2 followed by a call of SequenceRadix2 will multiply the
    17  // input sequence by the length of the sequence.
    18  //
    19  // CoefficientsRadix2 does not allocate, requiring that FFT twiddle factors
    20  // be calculated lazily. For performance reasons, this is done by successive
    21  // multiplication, so numerical accuracies can accumulate for large inputs.
    22  // If accuracy is needed, the FFT or CmplxFFT types should be used.
    23  //
    24  // If the length of seq is not an integer power of 2, CoefficientsRadix2 will
    25  // panic.
    26  func CoefficientsRadix2(seq []complex128) []complex128 {
    27  	x := seq
    28  	switch len(x) {
    29  	default:
    30  		if bits.OnesCount(uint(len(x))) != 1 {
    31  			panic("fourier: radix-2 fft called with non-power 2 length")
    32  		}
    33  
    34  	case 0, 1:
    35  		return x
    36  
    37  	case 2:
    38  		x[0], x[1] =
    39  			x[0]+x[1],
    40  			x[0]-x[1]
    41  		return x
    42  
    43  	case 4:
    44  		t := x[1] + x[3]
    45  		u := x[2]
    46  		v := negi(x[1] - x[3])
    47  		x[0], x[1], x[2], x[3] =
    48  			x[0]+u+t,
    49  			x[0]-u+v,
    50  			x[0]+u-t,
    51  			x[0]-u-v
    52  		return x
    53  	}
    54  
    55  	bitReversePermute(x)
    56  
    57  	for k := 0; k < len(x); k += 4 {
    58  		t := x[k+2] + x[k+3]
    59  		u := x[k+1]
    60  		v := negi(x[k+2] - x[k+3])
    61  		x[k], x[k+1], x[k+2], x[k+3] =
    62  			x[k]+u+t,
    63  			x[k]-u+v,
    64  			x[k]+u-t,
    65  			x[k]-u-v
    66  	}
    67  
    68  	for m := 4; m < len(x); m *= 2 {
    69  		f := swap(complex(math.Sincos(-math.Pi / float64(m))))
    70  		for k := 0; k < len(x); k += 2 * m {
    71  			w := 1 + 0i
    72  			for j := 0; j < m; j++ {
    73  				i := j + k
    74  
    75  				u := w * x[i+m]
    76  				x[i], x[i+m] =
    77  					x[i]+u,
    78  					x[i]-u
    79  
    80  				w *= f
    81  			}
    82  		}
    83  	}
    84  
    85  	return x
    86  }
    87  
    88  // bitReversePermute performs a bit-reversal permutation on x.
    89  func bitReversePermute(x []complex128) {
    90  	if len(x) < 2 || bits.OnesCount(uint(len(x))) != 1 {
    91  		panic("fourier: invalid bitReversePermute call")
    92  	}
    93  	lz := bits.LeadingZeros(uint(len(x) - 1))
    94  	i := 0
    95  	for ; i < len(x)/2; i++ {
    96  		j := int(bits.Reverse(uint(i)) >> lz)
    97  		if i < j {
    98  			x[i], x[j] = x[j], x[i]
    99  		}
   100  	}
   101  	for i++; i < len(x); i += 2 {
   102  		j := int(bits.Reverse(uint(i)) >> lz)
   103  		if i < j {
   104  			x[i], x[j] = x[j], x[i]
   105  		}
   106  	}
   107  }
   108  
   109  // SequenceRadix2 computes the real periodic sequence from the Fourier
   110  // coefficients, converting the frequency spectrum in coeff into a time
   111  // series, in place and returning it. This transform is unnormalized; a
   112  // call to CoefficientsRadix2 followed by a call of SequenceRadix2 will
   113  // multiply the input sequence by the length of the sequence.
   114  //
   115  // SequenceRadix2 does not allocate, requiring that FFT twiddle factors
   116  // be calculated lazily. For performance reasons, this is done by successive
   117  // multiplication, so numerical accuracies can accumulate for large inputs.
   118  // If accuracy is needed, the FFT or CmplxFFT types should be used.
   119  //
   120  // If the length of coeff is not an integer power of 2, SequenceRadix2
   121  // will panic.
   122  func SequenceRadix2(coeff []complex128) []complex128 {
   123  	x := coeff
   124  	for i, j := 1, len(x)-1; i < j; i, j = i+1, j-1 {
   125  		x[i], x[j] = x[j], x[i]
   126  	}
   127  
   128  	CoefficientsRadix2(x)
   129  	return x
   130  }
   131  
   132  // PadRadix2 returns the values in x in a slice that is an integer
   133  // power of 2 long. If x already has an integer power of 2 length
   134  // it is returned unaltered.
   135  func PadRadix2(x []complex128) []complex128 {
   136  	if len(x) == 0 {
   137  		return x
   138  	}
   139  	b := bits.Len(uint(len(x)))
   140  	if len(x) == 1<<(b-1) {
   141  		return x
   142  	}
   143  	p := make([]complex128, 1<<b)
   144  	copy(p, x)
   145  	return p
   146  }
   147  
   148  // TrimRadix2 returns the largest slice of x that is has an integer
   149  // power of 2 length, and a slice holding the remaining elements.
   150  func TrimRadix2(x []complex128) (even, remains []complex128) {
   151  	if len(x) == 0 {
   152  		return x, nil
   153  	}
   154  	n := 1 << (bits.Len(uint(len(x))) - 1)
   155  	return x[:n], x[n:]
   156  }
   157  
   158  // CoefficientsRadix4 computes the Fourier coefficients of the input
   159  // sequence, converting the time series in seq into the frequency spectrum,
   160  // in place and returning it. This transform is unnormalized; a call to
   161  // CoefficientsRadix4 followed by a call of SequenceRadix4 will multiply the
   162  // input sequence by the length of the sequence.
   163  //
   164  // CoefficientsRadix4 does not allocate, requiring that FFT twiddle factors
   165  // be calculated lazily. For performance reasons, this is done by successive
   166  // multiplication, so numerical accuracies can accumulate for large inputs.
   167  // If accuracy is needed, the FFT or CmplxFFT types should be used.
   168  //
   169  // If the length of seq is not an integer power of 4, CoefficientsRadix4 will
   170  // panic.
   171  func CoefficientsRadix4(seq []complex128) []complex128 {
   172  	x := seq
   173  	switch len(x) {
   174  	default:
   175  		if bits.OnesCount(uint(len(x))) != 1 || bits.TrailingZeros(uint(len(x)))&0x1 != 0 {
   176  			panic("fourier: radix-4 fft called with non-power 4 length")
   177  		}
   178  
   179  	case 0, 1:
   180  		return x
   181  
   182  	case 4:
   183  		t := x[1] + x[3]
   184  		u := x[2]
   185  		v := negi(x[1] - x[3])
   186  		x[0], x[1], x[2], x[3] =
   187  			x[0]+u+t,
   188  			x[0]-u+v,
   189  			x[0]+u-t,
   190  			x[0]-u-v
   191  		return x
   192  	}
   193  
   194  	bitPairReversePermute(x)
   195  
   196  	for k := 0; k < len(x); k += 4 {
   197  		t := x[k+1] + x[k+3]
   198  		u := x[k+2]
   199  		v := negi(x[k+1] - x[k+3])
   200  		x[k], x[k+1], x[k+2], x[k+3] =
   201  			x[k]+u+t,
   202  			x[k]-u+v,
   203  			x[k]+u-t,
   204  			x[k]-u-v
   205  	}
   206  
   207  	for m := 4; m < len(x); m *= 4 {
   208  		f := swap(complex(math.Sincos((-math.Pi / 2) / float64(m))))
   209  		for k := 0; k < len(x); k += m * 4 {
   210  			w := 1 + 0i
   211  			w2 := w
   212  			w3 := w2
   213  			for j := 0; j < m; j++ {
   214  				i := j + k
   215  
   216  				t := x[i+m]*w + x[i+3*m]*w3
   217  				u := x[i+2*m] * w2
   218  				v := negi(x[i+m]*w - x[i+3*m]*w3)
   219  				x[i], x[i+m], x[i+2*m], x[i+3*m] =
   220  					x[i]+u+t,
   221  					x[i]-u+v,
   222  					x[i]+u-t,
   223  					x[i]-u-v
   224  
   225  				wt := f
   226  				w *= wt
   227  				wt *= f
   228  				w2 *= wt
   229  				wt *= f
   230  				w3 *= wt
   231  			}
   232  		}
   233  	}
   234  
   235  	return x
   236  }
   237  
   238  // bitPairReversePermute performs a bit pair-reversal permutation on x.
   239  func bitPairReversePermute(x []complex128) {
   240  	if len(x) < 4 || bits.OnesCount(uint(len(x))) != 1 || bits.TrailingZeros(uint(len(x)))&0x1 != 0 {
   241  		panic("fourier: invalid bitPairReversePermute call")
   242  	}
   243  	lz := bits.LeadingZeros(uint(len(x) - 1))
   244  	i := 0
   245  	for ; i < 3*len(x)/4; i++ {
   246  		j := int(reversePairs(uint(i)) >> lz)
   247  		if i < j {
   248  			x[i], x[j] = x[j], x[i]
   249  		}
   250  	}
   251  	for i++; i < len(x); i += 2 {
   252  		j := int(reversePairs(uint(i)) >> lz)
   253  		if i < j {
   254  			x[i], x[j] = x[j], x[i]
   255  		}
   256  	}
   257  }
   258  
   259  // SequenceRadix4 computes the real periodic sequence from the Fourier
   260  // coefficients, converting the frequency spectrum in coeff into a time
   261  // series, in place and returning it. This transform is unnormalized; a
   262  // call to CoefficientsRadix4 followed by a call of SequenceRadix4 will
   263  // multiply the input sequence by the length of the sequence.
   264  //
   265  // SequenceRadix4 does not allocate, requiring that FFT twiddle factors
   266  // be calculated lazily. For performance reasons, this is done by successive
   267  // multiplication, so numerical accuracies can accumulate for large inputs.
   268  // If accuracy is needed, the FFT or CmplxFFT types should be used.
   269  //
   270  // If the length of coeff is not an integer power of 4, SequenceRadix4
   271  // will panic.
   272  func SequenceRadix4(coeff []complex128) []complex128 {
   273  	x := coeff
   274  	for i, j := 1, len(x)-1; i < j; i, j = i+1, j-1 {
   275  		x[i], x[j] = x[j], x[i]
   276  	}
   277  
   278  	CoefficientsRadix4(x)
   279  	return x
   280  }
   281  
   282  // PadRadix4 returns the values in x in a slice that is an integer
   283  // power of 4 long. If x already has an integer power of 4 length
   284  // it is returned unaltered.
   285  func PadRadix4(x []complex128) []complex128 {
   286  	if len(x) == 0 {
   287  		return x
   288  	}
   289  	b := bits.Len(uint(len(x)))
   290  	if len(x) == 1<<(b-1) && b&0x1 == 1 {
   291  		return x
   292  	}
   293  	p := make([]complex128, 1<<((b+1)&^0x1))
   294  	copy(p, x)
   295  	return p
   296  }
   297  
   298  // TrimRadix4 returns the largest slice of x that is has an integer
   299  // power of 4 length, and a slice holding the remaining elements.
   300  func TrimRadix4(x []complex128) (even, remains []complex128) {
   301  	if len(x) == 0 {
   302  		return x, nil
   303  	}
   304  	n := 1 << ((bits.Len(uint(len(x))) - 1) &^ 0x1)
   305  	return x[:n], x[n:]
   306  }
   307  
   308  // reversePairs returns the value of x with its bit pairs in reversed order.
   309  func reversePairs(x uint) uint {
   310  	if bits.UintSize == 32 {
   311  		return uint(reversePairs32(uint32(x)))
   312  	}
   313  	return uint(reversePairs64(uint64(x)))
   314  }
   315  
   316  const (
   317  	m1 = 0x3333333333333333
   318  	m2 = 0x0f0f0f0f0f0f0f0f
   319  )
   320  
   321  // reversePairs32 returns the value of x with its bit pairs in reversed order.
   322  func reversePairs32(x uint32) uint32 {
   323  	const m = 1<<32 - 1
   324  	x = x>>2&(m1&m) | x&(m1&m)<<2
   325  	x = x>>4&(m2&m) | x&(m2&m)<<4
   326  	return bits.ReverseBytes32(x)
   327  }
   328  
   329  // reversePairs64 returns the value of x with its bit pairs in reversed order.
   330  func reversePairs64(x uint64) uint64 {
   331  	const m = 1<<64 - 1
   332  	x = x>>2&(m1&m) | x&(m1&m)<<2
   333  	x = x>>4&(m2&m) | x&(m2&m)<<4
   334  	return bits.ReverseBytes64(x)
   335  }
   336  
   337  func negi(c complex128) complex128 {
   338  	return cmplx.Conj(swap(c))
   339  }
   340  
   341  func swap(c complex128) complex128 {
   342  	return complex(imag(c), real(c))
   343  }