何かあれば GitHub のリポジトリに issue を作るか ryukau@gmail.com までお気軽にどうぞ。
Update: 2025-01-07
AM (amplitude modulation) のアンチエイリアシングを行います。入力信号は音声です。
両側波帯は素朴な AM と等価。
以下は素朴な AM のブロック線図です。
以下は出力 \(y\) の項がすべて \(\sin\) になるようにした AM の式です。キャリアの振幅を 1 に正規化しています。
\[ \begin{aligned} x_c(t) &= \sin(\omega_c t), \\ x_m(t) &= m \cos(\omega_m t + \phi), \\ y(t) &= \big( B + x_m(t) \big) x_c(t) \\ &= B \sin(\omega_c t) + \frac{m}{2} \Big( \sin \big( (\omega_c + \omega_m) t + \phi \big) + \sin \big( (\omega_c - \omega_m) t - \phi \big) \Big). \end{aligned} \]
\(B\) と \(m\) によって変調の特性が変わります。参考にした Wikipedia の記事では \(B = 1\) としています。
ここで上側波帯の成分と、下側波帯の成分を以下のようにまとめます。
\[ \begin{aligned} U(t) &= \sin \big( (\omega_c + \omega_m) t + \phi \big). \\ L(t) &= \sin \big( (\omega_c - \omega_m) t - \phi \big). \\ \end{aligned} \]
\(U\) と \(L\) を使えば出力 \(y\) を以下のように書けます。
\[ y(t) = B x_c(t) + \frac{m}{2} \Big( U(t) + L(t) \Big). \]
素朴な AM は、以下で紹介するシングルサイドバンドと区別するために、ダブルサイドバンドと呼ばれることがあります。
キャリアとモジュレータを analytic signal に変換できれば上側波帯あるいは下側波帯のみを取り出すことができます。 Analytic signal は以下の式で計算できます。
\[ \mathtt{analytic} = x + j \mathcal{H}(x). \]
以下はシングルサイドバンド AM の計算式です。
\[ \begin{aligned} U(t) &= \mathrm{Re}(\hat{x}_c) \mathrm{Re}(\hat{x}_m) - \mathrm{Im}(\hat{x}_c) \mathrm{Im}(\hat{x}_m) = x_c x_m - \mathcal{H}(x_c) \mathcal{H}(x_m). \\ L(t) &= \mathrm{Re}(\hat{x}_c) \mathrm{Re}(\hat{x}_m) + \mathrm{Im}(\hat{x}_c) \mathrm{Im}(\hat{x}_m) = x_c x_m + \mathcal{H}(x_c) \mathcal{H}(x_m). \end{aligned} \]
\(U\) と \(L\) の式は analytic signal の定義より \(\mathrm{Re}(\hat{x}_c) \mathrm{Re}(\hat{x}_m) = x_c x_m\) と簡略化できます。しかし、リアルタイムで動作する analytic signal への変換を近似するフィルタ (Hilbert フィルタ) では実部の信号も変化するので、実装では簡略化する前の形を使うことがあります。 Hilbert フィルタについては「AM による周波数シフト」で実装例を紹介しています。
以下は scipy.signal.hilbert
を使った Python 3
によるシングルサイドバンド AM の実装です。
import scipy.signal as signal
import numpy as np
def upperAM(carrior, modulator):
= signal.hilbert(carrior)
c0 = signal.hilbert(modulator)
m0 return c0.real * m0.real - c0.imag * m0.imag
def lowerAM(carrior, modulator):
= signal.hilbert(carrior)
c0 = signal.hilbert(modulator)
m0 return c0.real * m0.real + c0.imag * m0.imag
以下はリアルタイム向けの実装へのリンクです。
音声信号は直流からナイキスト周波数までの全帯域が使われることが普通です。したがって AM をかけると全帯域に渡るエイリアシングが起きます。
以下は処理の各段階で現れる信号の周波数帯を示した図です。ここでのアンチエイリアシングは
\(\left[ -\dfrac{1}{2} f_s, 0
\right)\) と \(\left[ \dfrac{1}{2}
f_s, f_s \right)\) の成分を 0 にします。実装では rfft
を使うので負の周波数は扱っていません。
上側波帯は 1.5 倍のアップサンプリングを行った後に変調することでアンチエイリアシングができます。以下の図の \(\dfrac{3}{4} f_s\) が 1.5 倍のアップサンプリング後のナイキスト周波数です。赤で塗りつぶした部分が折り返しますが、ダウンサンプリングで \(\left[ \dfrac{1}{2} f_s, \dfrac{3}{4} f_s \right)\) の範囲を 0 にするので \(\left[ 0, \dfrac{1}{2} f_s \right)\) の範囲には折り返しません。
以下はリアルタイム向けの処理のブロック線図です。効率よく 1.5 倍のアップサンプリングを行う方法がわからないので 2 倍のアップサンプリングを使っています。
以下は Python 3 によるオフライン実装です。
def antiAliasedUpperAM(carrior, modulator):
= signal.resample(carrior, 2 * len(carrior))
upCar = signal.resample(modulator, 2 * len(modulator))
upMod return signal.resample(upperAM(upCar, upMod), len(carrior))
以下は signal.resample
をリアルタイムで動くハーフバンド楕円フィルタに置き換えた実装へのリンクです。
下側波帯は以下の手順でアンチエイリアシングができます。 \(f_N\) は元の信号のナイキスト周波数です。
周波数シフトについては「AM による周波数シフト」で手法を紹介しています。
以下はリアルタイム向けの処理のブロック線図です。
以下は Python 3 によるオフライン実装です。
def applySpectrumBandpass(sig, lowerCutoff, upperCutoff):
= np.fft.rfft(sig)
spectrum
# Highpass.
= int(len(spectrum) * 2 * lowerCutoff)
lowerIndex min(lowerIndex, len(spectrum))] = 0
spectrum[:
# Lowpass.
= int(len(spectrum) * 2 * upperCutoff)
upperIndex max(0, upperIndex) :] = 0
spectrum[
return np.fft.irfft(spectrum)
def frequencyShift(sig, normalizedFreq):
= signal.hilbert(sig)
analytic = np.abs(analytic)
norm = np.angle(analytic)
theta = np.linspace(0, len(analytic), len(analytic))
time return norm * np.cos(theta + 2 * np.pi * normalizedFreq * time)
def antiAliasedLowerAM(carrior, modulator):
= signal.resample(carrior, 2 * len(carrior))
upCar = signal.resample(modulator, 2 * len(modulator))
upMod
= frequencyShift(upCar, 0.25)
shiftedCar = lowerAM(shiftedCar, upMod)
am = applySpectrumBandpass(am, 0.25, 0.5)
filtered = frequencyShift(filtered, -0.25)
result
return signal.resample(result, len(carrior))
以下はリアルタイム向けの実装へのリンクです。
変調後にかけるハイパスフィルタは楕円フィルタのストップバンドゲインを高め (-140 dB ではなく -60 dB) に設定することでフォールオフを急峻にしています。人間の聴覚は低い周波数の解像度がいいので、低減の開始が 20 Hz 前後、折り返しは 100 Hz 以下でストップバンドゲインに到達するように設計しました。
素朴な AM は以下の手順でアンチエイリアシングができます。上側波帯と下側波帯が両方出るのでアップサンプリングの倍率を 3 倍にする点が異なります。上側波帯のアンチエイリアシングと同様に、理論上は 2.5 倍のアップサンプリングで十分ですが、実装では 3 倍にしたほうが楽です。
以下はリアルタイム向けの処理のブロック線図です。
以下は Python 3 によるオフライン実装です。
def antiAliasedAMFull(carrior, modulator):
= signal.resample(carrior, 3 * len(carrior))
upCar = signal.resample(modulator, 3 * len(modulator))
upMod
= frequencyShift(upCar, 1 / 6)
shiftedCar = shiftedCar * (1 + upMod)
am = applySpectrumBandpass(am, 1 / 6, 2 / 6)
filtered = frequencyShift(filtered, -1 / 6)
result
return signal.resample(result, len(carrior))
以下はリアルタイム向けの実装です。
以下にリンクした GitHub のリポジトリで AM のアンチエイリアシングを実装した AmplitudeModulator というプラグインのソースコードを配布しています。