#!/usr/bin/env perl
primal - generate notes at periodic intervals, the original being the
prime numbers 3 5 7 11 13 at some arbitrary pitch for each interval.
use 5.36.0;
use MIDI;
my $out_file = shift // 'out.midi';
my @beats = qw( 3 5 7 11 13);
my @pitch = qw(69 75 79 70 74);
How long to go on for; this might be some number of beats over some
number of measures (e.g. 64) or a least common multiple of the beats.
my $max_beat = 3466;
my $channel = 0; # MIDI channel
my $duration = 60; # minimum MIDI dtime
From the ##forth IRC chat:
< soweli_iki> they aren't primes, but you should try 8-12-15-18-20
sometime as well
if (1) {
@beats = qw( 8 12 15 18 20);
For better "mathcore" points we might map the beats to the pitches
(or even a frequency) by some scheme, e.g. by ratio or here to
modulate into the 12-tone pitch set, with an adjustment.
$ perl -E 'say for map { 60 + $_ % 12 } qw(8 12 15 18 20)'
...
$ atonal-util basic 68 60 63 66 80
c,d,f,aes
012111
4-27 Dominant seventh chord
c,ees,ges,aes half_prime
@pitch = qw(68 60 63 66 80);
The duration usually will need to be pretty long to hear "enough"
of the pattern variations, and might be automatically scaled by a
least common multiple, unless that value gets too large, unless
you're into that sort of thing.
$max_beat = 1024;
This will likely need to be fiddled with depending on the
beats involved.
$duration = 14;
}
my $beat = 0;
while ( $beat < $max_beat ) {
my @active;
for my $i ( 0 .. $#beats ) {
$active[$i] = $pitch[$i] if $beat % $beats[$i] == 0;
}
sound( $beat, $duration, @active ) if @active;
$beat++;
}
generate_midi();
{
my @last_beat; # when did the last note_on for the pitch fire?
my @trackevents; # array references of MIDI events for each pitch
sub generate_midi {
my $o = MIDI::Opus->new(
{ format => 1, # assume multiple tracks
ticks => 96,
tracks =>
[ map { MIDI::Track->new( { events => $_ } ) } @trackevents ]
}
);
$o->write_to_file($out_file);
}
Generate pitch-to-a-unique-track note events with appropriate
dtime values.
sub sound ( $beat, $dtime, @pitches ) {
for my $i ( 0 .. $#pitches ) {
next unless defined $pitches[$i];
my $previous = $last_beat[$i];
my $now = $beat * $dtime;
Start the MIDI events relative to the first note that sounds.
unless ( defined $previous ) {
$previous = $now;
@last_beat = ($now) x @last_beat;
}
my $offset = $now - $previous;
push @{ $trackevents[$i] },
[ note_on => $offset, $channel, $pitches[$i], velocity() ],
[ note_off => $dtime, $channel, $pitches[$i], 0 ];
Advance plus dtime due to the note_off consuming a beat.
To hold the note for longer one would need more dtime with
the note_off, at the risk of running into or beyond the
next note_on for the pitch.
$last_beat[$i] = $now + $dtime;
}
}
Humanize the loudness a little. Other options here would be to
impose some rhythmic pattern based on 4/4 or whatever time, or at
some period based on the interval the pitch sounds at (e.g. hit
harder every third note for 3, every 5th note for 5, etc). Or the
velocity could have a global adjustment based on a sine wave
unique to each beat, etc.
sub velocity { 80 + int( rand 6 + rand 6 + rand 6 ) }
}
Source