Generator form manipulation

Revision as of 23:35, 15 September 2021 by Cmloegcmluin (talk | contribs) (definition)

A canonical mapping form is an important standard to have as community for uniquely identifying temperaments, but it is not the only mapping form one should ever need, because one may wish to use differently-sized generators.

If two mappings are equivalent, i.e. they have the same canonical form and therefore represent the same temperament, then their corresponding generators are equivalent too. That doesn't mean their generators are the same sizes; it only means that in combination with each other, their generators reach the same set of pitches.

For example, the canonical form of 5-limit meantone is [1 1 0] 0 1 4], and form has generators with sizes of approximately an octave and a perfect fifth, respectively. But any pitch system constructed using an octave and a perfect fifth could also have been constructed using an octave and a perfect fourth, because the perfect fourth is the octave complement of the perfect fifth. Specifically, any pitch we reached previously with a perfect fifth could be instead reached by going up an octave and down a perfect fourth. So in situations where we're approaching 5-limit meantone as a pitch system constructed by an octave and a perfect fourth, we might prefer to have the mapping in that form, which looks like [1 2 4] 0 -1 -4].

As a further example, we might prefer only to use generators that are approximations of primes, so we'd like meantone's mapping in the form where the generators are an octave and a perfect twelfth, or tritave (3/1). This works for a similar reason: anything we could have reached with a perfect fifth we could also reach by moving up a tritave and down an octave, and that form looks like [1 0 -4] 0 1 4].

example meantone mapping forms
[⟨octave] ⟨fifth]⟩ [1 1 0] 0 1 4]
[⟨octave] ⟨fourth]⟩ [1 2 4] 0 -1 -4]
[⟨octave] ⟨tritave]⟩ [1 0 -4] 0 1 4]

Now clearly all three of these mapping forms look related, and they are indeed, but the exact relationships between them may not be immediately apparent, or how those relationships correspond to the relationships between their generator sizes. The purpose of this section is to demonstrate tricks for transforming from one matrix form to another so that we can make the generators the sizes we want, and along the way we'll look at how the tricks work in order to explain these relationships.

generator size manipulation tricks

first trick: change one generator by the size of another generator

The most basic trick is this: for a rank [math]\displaystyle{ r }[/math] temperament whose mapping has rows [math]\displaystyle{ R_1, R_2 ... R_r }[/math], and corresponding generators with cents [math]\displaystyle{ G_1, G_2 ... G_r }[/math], if we want to increase [math]\displaystyle{ G_a }[/math] by [math]\displaystyle{ G_b }[/math], then replace [math]\displaystyle{ R_b }[/math] with [math]\displaystyle{ R_b' = R_b - R_a }[/math].

Let's demonstrate this trick on the most recent example we looked at: meantone as generated by an octave and a tritave. If we begin with meantone in canonical form, [1 1 0] 0 1 4], where the generators are an octave and a perfect fifth, then we know we need to increase the second generator by the size of the first generator, because a tritave is equal to a perfect fifth plus an octave. So in terms of our variables, we must change [math]\displaystyle{ G_2 }[/math] so that it's [math]\displaystyle{ G_2 + G_1 }[/math]. According to our trick, then, we must replace [math]\displaystyle{ R_1 }[/math] with [math]\displaystyle{ R_1' = R_1 - R_2 }[/math]. And so [math]\displaystyle{ R_1' }[/math] is found like this:

[math]\displaystyle{ \begin{array} {r} R_1 \\ -R_2 \\ \hline R_1' \end{array} → \begin{array} {r} & \langle & 1 & 1 & 0 & ] \\ - & \langle & 0 & 1 & 4 & ] \\ \hline & \langle & 1 & 0 & -4 & ] \end{array} }[/math]

And then simply replace [math]\displaystyle{ R_1 }[/math] = 1 1 0] with [math]\displaystyle{ R_1' }[/math] = 1 0 -4] in the mapping, changing [1 1 0] 0 1 4] to [1 0 -4] 0 1 4].

It may be counterintuitive at first that in order to change the size of a generator we must make a change to a mapping row other than the one which corresponds to that generator (in this example, we changed the size of of the second generator by changing the first row). But there is a way to train our intuition on this effect. Think of it like transferring jobs. Remember that in order to know the size in cents of a generator, we cannot look exclusively at that generator's mapping row out of context; we need to know the entire mapping, because the mapping rows all work together to determine how the temperament works. So we can think of every mapping row like a workforce that outsources some of its work to the other rows; they're an interconnected system of workforces. And so when we subtract one row from another, we're in effect saying that the row being subtracted from is going to do less of the row being subtracted's work. So when we subtract [math]\displaystyle{ R_2 }[/math] from [math]\displaystyle{ R_1 }[/math], what we're saying is that whatever work the first generator was doing for the second generator, it will no longer do that work anymore, so the second generator will need to take care of that work itself; and that's why [math]\displaystyle{ G_2 }[/math] becomes the size of [math]\displaystyle{ G_2 }[/math] plus [math]\displaystyle{ G_1 }[/math].

Conversely, if we want to decrease [math]\displaystyle{ G_a }[/math] by [math]\displaystyle{ G_b }[/math], then we replace [math]\displaystyle{ R_b }[/math] with [math]\displaystyle{ R_b' = R_b }[/math] plus [math]\displaystyle{ R_a }[/math].

second trick: negating a generator

Now let's demonstrate a slightly more complicated example: transforming the 5-limit meantone mapping from its canonical form which uses a perfect fifth to the form which uses a perfect fourth. Again, we know this is possible because the perfect fourth is the octave complement of the perfect fifth.

This generator complementing principle is not exclusive to the octave; it is fully generalizable, which is to say that we may replace any generator with its complement with any other generator, and we'll still be describing the same pitch system. Again, this is because any pitch we could have reached previously we can still reach, by moving by a combination of the new generator and the other generator we just changed its size by[1].

We could describe replacing a generator with its complement with another generator like this: first negate the generator, and then add the other generator.

Therefore, our first step is to negate our meantone generator so that it no longer represents a perfect fifth upward, but rather a perfect fifth downward.

We can't accomplish this using the one trick we've already learned. How could we? That first trick relies on the interactions of two different generators, whereas negating a generator only involves one generator: itself. So here's where our second generator size manipulation trick comes in: negating a generator.

Fortunately this second trick is very easy. All we need to do in order to negate [math]\displaystyle{ G_a }[/math] (the generator size, e.g. in cents) is to negate each of the terms of the mapping row [math]\displaystyle{ R_a }[/math]. So, we would simply change [math]\displaystyle{ R_2 }[/math] = 0 1 4] to [math]\displaystyle{ R_2' }[/math] = 0 -1 -4], thereby changing the mapping from [1 1 0] 0 1 4] to [1 1 0] 0 -1 -4][2].

So at this point, the generators are an octave and a negative perfect fifth. It's time for our second step.

The second step of achieving a generator with the size of a perfect fourth should be familiar: we need to increase our negative perfect fifth to a perfect fourth, and so we need to add one octave, and therefore we need to change [math]\displaystyle{ R_1 }[/math] to be [math]\displaystyle{ R_1 - R_2 }[/math], so we end up with:

[math]\displaystyle{ \begin{array} {r} R_1 \\ -R_2 \\ \hline R_1' \end{array} → \begin{array} {r} & \langle & 1 & 1 & 0 & ] \\ - & \langle & 0 & -1 & -4 & ] \\ \hline & \langle & 1 & 2 & 4 & ] \end{array} }[/math]

And we're done, having found the mapping [1 2 4] 0 -1 -4].

beyond rank-2

 
A demonstration of how one might transform the size of a generator of a rank-3 temperament.

These two tricks should enable us to attain any valid generator sizes we may wish for a given temperament. And these tricks work for any rank[3], not only rank 2 like we've looked at thus far with 5-limit meantone examples. To be clear, for a rank [math]\displaystyle{ r }[/math] temperament, adding or subtracting [math]\displaystyle{ R_a }[/math] from another generator will only affect the size of [math]\displaystyle{ G_a }[/math].

For example, 7-limit marvel's canonical form is [1 0 0 -5] 0 1 0 2] 0 0 1 2], with generators of an octave, tritave, and pentave (5/1), in that order. We can change that second generator from a tritave to a perfect fifth by decreasing [math]\displaystyle{ G_2 }[/math] by [math]\displaystyle{ G_1 }[/math], which we know by the first trick means we add [math]\displaystyle{ R_2 }[/math] to [math]\displaystyle{ R_1 }[/math], producing [1 1 0 -3] 0 1 0 2] 0 0 1 2]. Helpfully, this trick has no effect on any other generators that were not involved, which in this case is just the size of the pentave, which was the one remaining generator out of the three in this temperament.

The fact that these tricks have isolated effects on the generator sizes like this makes it straightforward to compose sequences of them, applied one after the other, to attain an incredible variety of valid generator sizes, as you can see in the diagram to the right.

mingen form

definition

A number of generator size targets have been described, such as chroma-positive generators, "musician's form", and probably more will be described in the future[4]. The generator size target we'll be demonstrating here may not have been rigorously described elsewhere, or named[5], but these purposes, we'll be calling it minimum-generator, or mingen, for short. Mingen form is when each successive generator is less than half the size of the previous generator.[6][7]

In order to define the size of the generators, you need to specify a tuning strategy. Though if you have two generators that are close enough that their size ranking depends on the tuning, then you probably have other problems. In any case, the tuning strategy that we'll be using here is T2, because it's decent enough and easy to compute.

rank-2 instructions table

Let's begin with the simple example of 5-limit meantone. The following table demonstrates how to obtain mingen form from various starting positions, by synthesizing the two generator size manipulation tricks explained in the previous section.

mapping manipulations to where where period [math]\displaystyle{ p }[/math] is the first mapping row r₁ in cents and generator [math]\displaystyle{ g }[/math] is the second mapping row r₂ in cents
example matrix example cents current [math]\displaystyle{ p }[/math] and [math]\displaystyle{ g }[/math] desired new [math]\displaystyle{ g }[/math] required [math]\displaystyle{ r_1 }[/math] change required r₂ change repeat?
{{1,0,-4},{0,-1,-4}} {1201.4,-1898.4} g < −p g + p r₁ − 2r₂ yes
{{1,1,0},{0,-1,-4}} {1201.4,-697.049} −p <= g < −p/2 p + g r₁ − r₂ no, you're done
{{1,2,4},{0,1,4}} {1201.4,-504.348} −p/2 <= g < 0 −g −r₂ no, you're done
{{1,2,4},{0,-1,-4}} {1201.4,504.4} 0 <= g <= p/2 g no, you're done
{{1,1,0},{0,1,4}} {1201.4,697.049} p/2 < g <= p p - g r₁ + r₂ −r₂ no, you're done
{{1,0,-4},{0,1,4}} {1201.4,1898.4} p < g g - p r₁ + 2r₂ −r₂ yes

Wolfram Language implementation

The below code essentially works through the input matrix M two rows at a time, beginning with the first two rows. Each pair of rows is "fixed" so that the second row is less than half of the first row. The same set of changes and potential recursions as described in the table in the previous section is used for each pair of rows. When the sizes of the generators is computed, a weighted Frobenius tuning is used for its computational frugality and reasonableness.

jip[d_] := Map[Log2, Map[Prime,Range[d]]]
tenneyWeights[m_] := DiagonalMatrix[1/jip[Last[Dimensions[m]]]]
unweightedG[m_] := PseudoInverse[m]
weightedG[m_] := tenneyWeights[m].PseudoInverse[m.tenneyWeights[m]] // N
octaves[vOrVList_]:= N[jip[Length[vOrVList]].vOrVList, 10]
cents[vOrVList_] := N[1200.octaves[vOrVList], 10]

gens[m_] := cents[weightedG[m]]


fixM[m_, i_] := Module[{localM, rp, rg, mGens, p, g, fixedRpRg, fixedM},
	localM = m;
	rp = m[[i]];
	rg = m[[i+1]];

	mGens = gens[m];
	p = mGens[[i]];
	g = mGens[[i+1]];

	fixedRpRg = Which[
		g < -p, {rp-2*rg, rg},
		-p <= g < -p/2, {rp-rg, rg},
		-p/2 <= g < 0, {rp, -rg},
		0 <= g <= p/2, {rp, rg},
		p/2 < g <=p, {rp+rg, -rg},
		p < g, {rp+2*rg, -rg}
	];

	fixedM = Take[localM, i-1] ~ Join ~ fixedRpRg ~ Join ~ Drop[localM, i+1];

	If[
		g < -p || p < g,
		fixM[fixedM, i],
		fixedM
	]
]

mingen[m_] := Module[{localM, rp, rg, fixedM},
	localM = m;

	For[i = 1, i < Length[localM], i++,
		localM = fixM[localM, i];
	];

	localM
]


(* examples *)

tester[m_] := Module[{ming},
	ming = mingen[m];
	"in: " <> ToString@m <> " w/ gens: " <> ToString@gens[m] <> "\nout: " <> ToString@ming <> " w/ gens: " <> ToString@gens[ming]
]

tester[{{1, 0, -4}, {0, -1, -4}}]
tester[{{1, 1, 0}, {0, -1, -4}}]
tester[{{1, 2, 4}, {0, 1, 4}}]
tester[{{1, 2, 4}, {0, -1, -4}}]
tester[{{1, 1, 0}, {0, 1, 4}}]
tester[{{1, 0, -4}, {0, 1, 4}}]


tester[{{12, 19, 28}}]
tester[{{1, 2, 3}, {0, 3, 5}}]
tester[{{1, 0, -4, -13}, {0, 1, 4, 10}}]
tester[{{5, 8, 0}, {0, 0, 1}}]
tester[{{2, 0, 11, 12}, {0, 1, -2, -2}}]
tester[{{1, 0, 0, -5}, {0, 1, 0, 2}, {0, 0, 1, 2}}]
tester[{{1, 0, 0, -5, 12}, {0, 1, 0, 2, -1}, {0, 0, 1, 2, -3}}]
tester[{{1, 8, 0}, {0, 11, -4}}]


dimensionality = RandomInteger[{3, 6}];
rank = RandomInteger[{2, dimensionality}];
randomM = RandomInteger[{-10, 10}, {rank, dimensionality}]
tester[randomM]

limitations

Beyond rank-2, the mingen form of a temperament is no longer unique. You can always get smaller and smaller generators.[8]

Consider the example in the diagram above. We begin with [1 2 0 -1] 0 -1 6 10] 0 0 -1 -2] with generators of 1200.6¢, 499.841¢, and 214.024¢, which therefore already satisfies the condition that each generator is less than half the previous generator. But we can transform it into [1 2 2 3] 0 -1 1 0] 0 0 -1 -2] which has a third generator of 116.013¢ instead. This is accomplished by adding row 3 to row 2 five times, which decreases generator 3 by the size of five times row 2, from 214.024¢ by 5 × 499.841 = 2499.205¢ to -2285.18¢; and then subtracting row 3 from row 1 twice, which increases generator 3 by the size of two times row 1, from -2285.18¢ by 2 × 1200.6¢ = 2401.2¢ to 116.013¢. And we can get that generator even smaller if we had instead moved up by 499.841 twice to 1213.71¢ and then down by 1200.6¢ once to 13.109¢ (that's a final mapping of [1 2 -1 -3] 0 -1 8 14] 0 0 -1 -2].

You could find smaller and smaller generators if you wanted, by essentially finding increasingly small "commas" between the other generators' sizes (e.g. 5 × 1200.6¢ versus 12 × 499.841¢ is a difference of only 4.908¢) and then shifting generators by those commas.

References

  1. And, of course, we can't reach any new pitches we couldn't already reach previously, either.
  2. To extend the workplace analogy (if we really like, though I doubt it's helpful here), we could say that we changed R_2's jobs so that they now do the exact opposite of what they used to do, so if we want them to accomplish the same thing as they used to, we have to have them undo their work.
  3. past 1, anyway; of course, rank 1 temperaments are somewhat inflexible in their single generator's size.
  4. Such as forms using LLL to get "a simple set of generators", per a post by Graham Breed on Facebook here.
  5. It is the form used by Graham Breed's regular temperament finding scripts hosted here: http://x31eq.com/temper/net.html, at least for rank-2 temperaments.
  6. Like octave reduction combined with octave inversion, since you can't just add or subtract half octaves until it's between 0 and 600 cents. You have to add or subtract octaves until it's between -600 and +600 cents, then multiply by -1 if it's negative.
  7. You could always find a smaller and smaller generator by going negative, so this assumes positive generators.
  8. This is why on Graham Breed's temperament finding tool, beyond rank-2 he simply uses the Hermite Normal Form.