Tuning Systems#

import IPython.display as ipd
import matplotlib.pyplot as plt
import numpy
import scipy
import scipy.signal

from mirdotcom import mirdotcom

mirdotcom.init()

Introduction#

Tuning Systems#

In twelve-tone equal temperament (Wikipedia), all twelve semitones within the octave have the same width. With this tuning system, expressed as a frequency ratio, the interval of one semitone is \(2^{1/12}\). Expressed in cents, this same interval is defined to be 100 cents. Therefore, the octave has 1200 cents.

In just intonation (Wikipedia), the frequency ratio is expressed as a fraction between two small integers, e.g. 3:2, 4:3. As a result, the higher harmonic partials between two notes will overlap, resulting in a consonant interval that is pleasing to the ear. In 5-limit just tuning, these fractions are expressed with prime factors no larger than 5, i.e. {2, 3, 5}. In 7-limit just tuning, these fractions are expressed with prime factors no larger than 7, i.e. {2, 3, 5, 7}. For example, 7:4 is a 7-limit interval, but it is not a 5-limit interval.

In Pythagorean tuning (Wikipedia), every frequency ratio is based upon the ratio 3:2. To find that ratio, from one note in the interval, step around the Circle of Fifths until you reach the other note in the interval, multiplying (if stepping forward) or dividing (if stepping backward) by 3/2 with each step. Finally, multiply or divide by 2 enough times to return to the octave of interest. Pythagorean tuning can also be considered 3-limit just tuning since every ratio only uses prime factors no greater than 3.

How To Use This Notebook#

In the examples below, listen to each interval, and compare intervals both visually and aurally across tuning systems. In some tuning systems, the upper harmonics do not align. In such cases, try to listen for dissonance and beat frequencies. Then compare them to the tuning systems where the harmonics align perfectly.

Notebook Setup#

Global parameters for this notebook:

T = 5  # duration in seconds
sr = 22050  # sampling rate in Hertz
fcutoff = 4000  # frequency cutoff for filter

Functions used to create the sounds and figures:

def simulate_tone(f0, harmonics=None):
    """Returns a tone with a specified fundamental frequency and harmonic amplitudes."""
    if harmonics is None:
        harmonics = [1]
    t = numpy.linspace(0, T, T * sr, endpoint=False)
    x = sum(
        v * numpy.sin(2 * numpy.pi * i * f0 * t) for i, v in enumerate(harmonics, 1)
    )
    return x
def filter_tone(x, fcutoff=None):
    """Return a low-pass filtered signal."""
    if fcutoff is None:
        fcutoff = sr / 2.0
    h = scipy.signal.firwin(55, 2 * float(fcutoff) / sr)
    y = scipy.signal.convolve(x, h)
    return y
def make_double_stop(ratio, f0=440, harmonics=None):
    """Listen to two tones played simultaneously, and plot the tones' spectra."""
    if harmonics is None:
        harmonics = [1]
    f1 = ratio * f0

    # Generate both tones.
    x0 = filter_tone(simulate_tone(f0, harmonics=harmonics), fcutoff=fcutoff)
    x1 = filter_tone(simulate_tone(f1, harmonics=harmonics), fcutoff=fcutoff)

    # Add both tones, and normalize.
    y = x0 + x1
    y = 0.5 * y / y.max()

    # Generate both spectra.
    X0 = scipy.fft.fft(x0)
    X1 = scipy.fft.fft(x1)

    # Create frequency variable.
    N = len(X0)
    f = numpy.linspace(0, sr, N, endpoint=False)

    # Plot spectrum of both notes.
    plt.semilogy(f, abs(X0), marker="o", linewidth=1)
    plt.semilogy(f, abs(X1), color="r", linewidth=1)

    plt.xlim(xmin=0, xmax=(len(harmonics) + 1) * (f1 if ratio > 1 else f0))
    plt.ylim(ymin=0.1)

    plt.xlabel("Frequency (Hz)")
    plt.ylabel("Magnitude")
    plt.legend(("f0 = %.2f Hz" % f0, "f0 = %.2f Hz" % f1))

    # Output audio widget.
    print("ratio between notes:", ratio)
    print("difference in cents:", numpy.log2(ratio) * 1200)
    return ipd.Audio(y, rate=sr)

Unison#

Within each section below, the intervals are provided in order of interval width from lowest to highest.

make_double_stop(1)
ratio between notes: 1
difference in cents: 0.0
../../_images/2b7d66d4365a1a952d17287a7fffec0ac7158630ffc56e7d26b33bf2fcd81476.png

Octaves#

harmonics = [1.0, 0.5]

Just intonation or equal temperament, 12 semitones, 1200 cents:

make_double_stop(2, harmonics=harmonics)
ratio between notes: 2
difference in cents: 1200.0
../../_images/a5996d8b3154bb9f47e25b9eca6c0507e67ed4aa06aa3eeb9d1d0a8f40c61141.png

Pythagorean tuning, twelve steps forward on the Circle of Fifths. In Pythagorean tuning, multiply the fundamental frequency by 3/2 twelve times, and then divide by two enough times to return to the octave of interest:

\[ \left( \frac{3}{2} \right)^{12} \left( \frac{1}{2} \right)^6 = \frac{531441}{262144} \approx 2.0273 \]
make_double_stop(531441.0 / 262144, harmonics=harmonics)
ratio between notes: 2.0272865295410156
difference in cents: 1223.4600103846492
../../_images/cce6ce7a81e939c33553bac6dfa59d1fe12f4b56ca8d52b94732ef9aa04dc0b2.png

The Pythagorean comma, the degree of inconsistency when trying to define a twelve-tone scale using only perfect fifths, is about 1.0136 when expressed as a frequency ratio:

\[ \left( \frac{3}{2} \right)^{12} \left( \frac{1}{2} \right)^7 = \frac{531441}{524288} \approx 1.0136 \]

Fifths#

harmonics = [1.0, 0.1, 0.1]

Equal temperament, seven semitones, 700 cents:

make_double_stop(2 ** (7.0 / 12), harmonics=harmonics)
ratio between notes: 1.4983070768766815
difference in cents: 700.0
../../_images/0bb93c75d0299d950881557293449740825d98c86d66906a0e04c8fc2ed77bfe.png

Just intonation or Pythagorean tuning, one step forward on the Circle of Fifths:

make_double_stop(3.0 / 2, harmonics=harmonics)
ratio between notes: 1.5
difference in cents: 701.9550008653874
../../_images/f913b21458f7483125ad2acb292b45e0f9e4d8632ebbcbca04b50ea2c77c20fa.png

Fourths#

harmonics = [1.0, 0.01, 0.1, 0.1]

Just intonation or Pythagorean tuning, one step backward on the Circle of Fifths:

\[ \left( \frac{2}{3} \right) \left( \frac{2}{1} \right) = \frac{4}{3} = 1.\overline{3} \]
make_double_stop(4.0 / 3, harmonics=harmonics)
ratio between notes: 1.3333333333333333
difference in cents: 498.0449991346125
../../_images/3c0970bc13a31e27488b8f8b8483e66b7bd72c4354b8f3b443745903b17c22fd.png

Equal temperament, five semitones:

make_double_stop(2 ** (5.0 / 12), harmonics=harmonics)
ratio between notes: 1.3348398541700344
difference in cents: 500.0
../../_images/a5d7bb85e5a1bec5dc49cbe66f60da60d9ec02b67653ef0d70a23c410ab70435.png

Major Sixths#

harmonics = [1.0, 0.01, 0.1, 0.01, 0.1]

Just intonation:

make_double_stop(5.0 / 3, harmonics=harmonics)
ratio between notes: 1.6666666666666667
difference in cents: 884.3587129994474
../../_images/989cca7089fe2b41a472aa650815e331fa07979ee0e218944ee74c276164ede5.png

Equal temperament, i.e. nine semitones:

make_double_stop(2 ** (9.0 / 12), harmonics=harmonics)
ratio between notes: 1.681792830507429
difference in cents: 899.9999999999999
../../_images/7a362bf4491c8b6efedc4f743813f3656408b32d58d2dcddab77e68a479838ed.png

Pythagorean tuning, three steps forward on the Circle of Fifths:

\[ \left( \frac{3}{2} \right)^3 \left( \frac{1}{2} \right) = \frac{27}{16} = 1.6875 \]
make_double_stop(27.0 / 16, harmonics=harmonics)
ratio between notes: 1.6875
difference in cents: 905.8650025961623
../../_images/c351d241981b7a0c1a0935a95197edf948c326de16d14350260335920fa5082b.png

Minor Thirds#

harmonics = [1.0, 0.01, 0.01, 0.01, 0.1, 0.1]

Pythagorean tuning, three steps backward on the Circle of Fifths:

\[ \left( \frac{2}{3} \right)^3 \left( \frac{2}{1} \right)^2 = \frac{32}{27} = 1.\overline{185} \]
make_double_stop(32.0 / 27, harmonics=harmonics)
ratio between notes: 1.1851851851851851
difference in cents: 294.13499740383764
../../_images/e7c12dff2c82552c1f1fe607426a3937876501c373c46f4556b905b1e6fad638.png

Equal temperament, three semitones:

make_double_stop(2 ** (3.0 / 12), harmonics=harmonics)
ratio between notes: 1.189207115002721
difference in cents: 299.99999999999994
../../_images/8213b9594c83000417bbd7f4c309844438cc2d5f315a32d8400beb30324332ef.png

Just intonation:

make_double_stop(6.0 / 5, harmonics=harmonics)
ratio between notes: 1.2
difference in cents: 315.64128700055255
../../_images/ef3d066445538b89b7123aa7d391a45f4970161485f42707721992e24bac807a.png

Major Thirds#

harmonics = [1.0, 0.01, 0.01, 0.1, 0.1]

Just intonation:

make_double_stop(5.0 / 4, harmonics=harmonics)
ratio between notes: 1.25
difference in cents: 386.3137138648348
../../_images/45c25c8f29f97cefee6af2f3d806c78afa6286452e7b0bfdc894667144ecaa78.png

Equal temperament, four semitones:

make_double_stop(2 ** (4.0 / 12), harmonics=harmonics)
ratio between notes: 1.2599210498948732
difference in cents: 400.00000000000006
../../_images/e99839db952cc6cbf3aa5f486cdb5ed31ed1bc8391135f97dd19b174092d07d1.png

Pythagorean tuning, four steps forward on the Circle of Fifths:

\[ \left( \frac{3}{2} \right)^4 \left( \frac{1}{2} \right)^2 = \frac{81}{64} = 1.265625 \]
make_double_stop(81.0 / 64, harmonics=harmonics)
ratio between notes: 1.265625
difference in cents: 407.8200034615497
../../_images/696acd62f01003b34968a77d1e3f245cad80301c02192719f32576623fe44318.png

Minor Sixths#

harmonics = [1.0, 0.001, 0.001, 0.001, 0.01, 0.001, 0.001, 0.01]

Pythagorean tuning, four steps backward on the Circle of Fifths:

\[ \left( \frac{2}{3} \right)^4 \left( \frac{2}{1} \right)^3 = \frac{128}{81} \approx 1.58 \]
make_double_stop(128.0 / 81, harmonics=harmonics)
ratio between notes: 1.5802469135802468
difference in cents: 792.1799965384502
../../_images/ff97eaa23b4572fa139be915d619381384268cdcca144a87828bb2147fb0d704.png

Equal temperament, eight semitones:

make_double_stop(2 ** (8.0 / 12), harmonics=harmonics)
ratio between notes: 1.5874010519681994
difference in cents: 799.9999999999998