-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathFMODDOC.TXT
2777 lines (2248 loc) · 107 KB
/
FMODDOC.TXT
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
/* (tabstops=8)
°°°°°°°°°°°°°°±±±±±±±±±±±±±²²²²²²²²²²²²²²²²²²²²²²²²±±±±±±±±±±±±±°°°°°°°°°°°°°°
þ MOD Player Tutorial by FireLight þ Copyright (c) Brett Paterson 1994-95 þ
þ Last updated 17/11/95 þ
°°°°°°°°°°°°°°±±±±±±±±±±±±±²²²²²²²²²²²²²²²²²²²²²²²²±±±±±±±±±±±±±°°°°°°°°°°°°°°
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² : SECTION 0: ²±° ³
³ °±² Index ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
Section 1 : INTRODUCTION
1.1 New in this version
1.2 Notes
1.3 Terminology
1.4 Contacting FireLight and feedback
Section 2 : THE LOADER
2.1 Notes
2.2 Verification
2.3 Load Module Name
2.4 Load Sample Information
2.5 Load Order Information
2.6 Load Pattern Data
2.6.1 Four bytes?
2.7 Load Sample Data
2.8 Phew :)
Section 3 : PLAYING THE MOD
3.1 Ok Where Do I Start?
3.1.1 Speed and BPM
3.2 Setting The Timer's Speed
3.3 Player Logic
3.3.1 Orders/Patterns
3.3.2 The Pattern Buffer
3.4 Inside Update Row
3.5 Period Frequencies and Fine Tune
3.5.1 What do I do with this table?
3.5.2 Gravis UltraSound
3.6 Volume
Section 4 : MISCELLANEOUS
4.1 Notes Without Instrument Numbers or Frequencies
4.2 Effect Bxy and Dxy on the same row - backwards mods
4.3 Logic with Breaking and Jumping - pointers, orders, rows
Section 5 : EFFECTS
5.1 Effect 0xy (Arpeggio)
5.2 Effect 1xy (Porta Up)
5.3 Effect 2xy (Porta Down)
5.4 Effect 3xy (Porta To Note)
5.5 Effect 4xy (Vibrato)
5.6 Effect 5xy (Porta + Vol Slide)
5.7 Effect 6xy (Vibrato + Vol Slide)
5.8 Effect 7xy (Tremolo)
5.9 Effect 8xy (Pan)
5.10 Effect 9xy (Sample Offset)
5.11 Effect Axy (Volume Slide)
5.12 Effect Bxy (Jump To Pattern)
5.13 Effect Cxy (Set Volume)
5.14 Effect Dxy (Pattern Break)
5.15 Effect Fxy (Set Speed)
5.16 Effect E0x (Set Filter)
5.17 Effect E1x (Fine Porta Up)
5.18 Effect E2x (Fine Porta Down)
5.19 Effect E3x (Glissando Control)
5.20 Effect E4x (Set Vibrato Waveform)
5.21 Effect E5x (Set Finetune)
5.22 Effect E6x (Pattern Loop)
5.23 Effect E7x (Set Tremolo WaveForm)
5.24 Effect E8x (Unused)
5.25 Effect E9x (Retrig Note)
5.26 Effect EAx (Fine Volume Slide Up)
5.27 Effect EBx (Fine Volume Slide Down)
5.28 Effect ECx (Cut Note)
5.29 Effect EDx (Delay Note)
5.30 Effect EEx (Pattern Delay)
5.31 Effect EFx (Invert Loop)
Section 6 : APPENDIX - MOD FORMAT DOCUMENT
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² : SECTION 1: ²±° ³
³ °±² Introduction ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² 1.1 New in this version ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
New:
====
- Section 3.1.1 - Speed and BPM.
- Section 3.3.2 - The Pattern Buffer
- Section 4.2 - Effect Bxy and Dxy on the same row - backwards mods
Changed/Added too:
==================
- Section 2.6 - Load pattern data section REWRITTEN - better pattern storage
methods are discussed, along with memory handling.
- Section 2.6.1 - 4 bytes? section changed and ADDED too. - new asm section.
- Section 3.2 - Setting the timer speed section expanded and is clearer.
- Section 3.4 - Inside update row pseudocode FIXED and REWRITTEN.
- Section 3.5.2 - GUS frequency info completely changed, new GUS volume table
- Section 5.1 - arpeggio, fixed (you add 8*finetune's not just 1 finetune)
it is also added too for more understanding.
- Section 5.5/ - vibrato/tremolo section totally REWRITTEN - DOUBLED in size.
5.8 it's now fully comprehensive and easy to understand.
- Section 5.12/ - pattern jump/pattern break added too.
5.14
- Section 5.19 - Effect E3x, Glissando added too, includes searching info.
- Section 5.20 - wavecontrol explained better, and ADDED to.
- Section 5.22 - Effect E6x, pattern loop bugfixed
- Section 5.29 - Effect EDx, delay note added too, warnings about quirks
- Spellchecked, and many other minor alterations/changes.
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² 1.2 Notes ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
I know half of you out there want this document to write a player for you,
but I have just got to say that this is a REFERENCE, and even though I have
done my utter best I can to provide you with a head start in writing a mod
player, nothing beats experimentation and writing stupid mods, and playing
them on your player to see if you did it right.
I could lay down more rules and say "this is how you do it" but this is
exactly what I DON'T want to do.
Assumptions:
============
Throughout the document, exaggerated length variable names are used, I don't
actually use these sort of variable names but they help to make things
clearer. e.g. "NUMBER_OF_PATTERNS". Variable names will be all stated in
capitals.
It is assumed you will have some sort of knowledge about
- Sound Cards (and programming of sound cards, though I do include GUS
code in the files GUS.ASM/GUS.CPP, and info on the Sound
Blaster is contained in FSBDOC.TXT)
- Interrupt Handlers (I will cover this though)
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² 1.3 Terminology ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
TYPE LENGTH BITS RANGE WATCOM/BORLAND/TURBO C
ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
byte 1 8 0-255 unsigned char
word 2 16 0-65,535 unsigned short
dword 4 32 0-4,294,967,295 unsigned long
ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
Throughout this text I use the terms BYTE,WORD, and DWORD, to make the
document more general to all languages. In C you can use typedefs to achieve
the use of byte,word,dword terminology, and in pascal and asm the syntax is
already suited to this anyway.
ORDERS - orders are how the mod plays from 0 to length of song.
PATTERNS - patterns are played in any ORDER, and are the physical information.
TICK - I refer to a clock tick for the interrupt handler as a tick, some
others use the term FRAME. I will be using the term tick throughout the whole
document.
ALL of the time I present a type of pseudocode to try not to seem too biased
towards a language, and rarely will I used straight C code to demonstrate how
it is done.
I ALWAYS explain it properly in english and pseudocode first, before I
resort to any real code though.
In the pseudocode the following terms are used:
===============================================
x SHR y = shift x right by y bits
x SHL y = shift x left by y bits
x AND y = bitwise AND x by y
x OR y = bitwise OR x by y
When I do use C to display how my code actually works, then here are some
of the symbols and what they mean for the C illiterate:
=======================================================
>> = bitwise shift right
<< = bitwise shift left
& = bitwise AND
| = bitwise OR
&& = logical AND
|| = logical OR
% = modulus operator (remainder)
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² 1.4 Contacting FireLight and feedback ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
email : [email protected]
post : Brett Paterson,
48/a Parr st,
Leongatha, 3953,
Victoria, Australia.
IRC : FireLight on #coders, #trax or #aussies
My internet access is limited at the moment so I might not be able to reply,
but who knows I might have proper access by the time you read this.
I have spent about a YEAR compiling and writing these documents, and it would
be encouraged if you could show your support by sending me a postcard to the
above postal address.
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² : SECTION 2 : ²±° ³
³ °±² The Loader ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² 2.1 Notes ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
Well first we've got to load the module in right? Following is a step by
step way to code your loader, and storage issues will be discussed to help
you along. I really don't feel like just writing another MOD format
description, so you will find one in the appendix of section 6 written by
Lars Hamre(?), the author of Protracker.
You WILL need to refer to the format document and this document side by side.
The loader section of this document doesn't actually give a map of mod format
and could be confusing, though it does go through it byte by byte.
The following section has their subsections which are in boxes, and in each
of these sections are 4 important subsections
- EXPLANATION (describes what the section is on about, for understanding)
- PSEUDOCODE (actually shows HOW to load the information)
- STORAGE ISSUE (helps on how to store the information loaded)
- SUGGESTION (a helpful hint or suggestion to do after this step)
I placed the pseudocode section before storage issues because I know you are
probably going to be eager and want to jump into some code straight away.
Storage issue follows just to be a guiding hand; not a 'must'.
each pseudocode section follows on from the last.
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² 2.2 Verification ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
Explanation:
============
Before we attempt to load a mod, we should check that it is in fact a mod.
Every mod has a unique signature, and in case of the .MOD format, this is
in the form of a 4 letter string containing the letters "M.K.", or "8CHN" or
a variety of other signatures for their mutated formats.
These describe the type of mod, and the identifier signature is stored at
offset 1080 (438h) in the file, so should be checked first.
PseudoCode:
===========
- Seek to offset 1080 (438h) in the file
- read in 4 bytes
- compare them to "M.K." - if true we have a 4 channel mod
- otherwise compare them to "6CHN" - if true we have a 6 channel mod
- otherwise compare them to "8CHN" - if true we have an 8 channel mod
- otherwise exit and display error message.
There are also rare tunes that use **CH where ** = 10-32 channels
Suggestion:
===========
Use this point to store the number of channels in a variable of your choice
(I just use a global variable called CHANNELS)
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² 2.3 Load Module Name ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
Explanation:
============
This is a trivial part of the loader and just holds the Title or name of the
mod. It is the very first 20 bytes of the MOD.
PsuedoCode:
===========
- Seek back to position 0, the start of the file
- read in 20 bytes, store as MODULE_NAME.
Storage Issue:
==============
The name of the module is a 20 byte string, padded by 0's.
Here you can either store your module name as a global variable, in a
character string, or do what I do and store all the general information about
the mod in a structure like this
struct MODHEADER {
char NAME[20]
...
other information (will get to this later)
...
} MODHEAD
OR just
char NAME[20]
It's a good idea to set up a structure like this for future use, there is a
lot more information we will need to throw in here later, but of course you
don't need a structure, you can keep it as a heap of loose variables.
And of course if you are not interested in displaying the name of the module
you could just discard it.
Suggestion:
===========
Code a 1 line program to print out the name of your module to see if it's
working properly. (exciting huh)
NOTE: The Module name is supposed to be padded by 0's, and terminated with a
0, but sometimes this is not the case. Sometimes a tracker will allow
all 20 bytes to store characters, which means no NULL terminator byte.
This causes functions like printf to give unpredictable output as it
cannot find the NULL terminator. The way to fix this is just to use a
loop and print out each character one at a time, or overwrite the 20th
byte with a 0.
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² 2.4 Load Sample Information ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
Explanation:
============
Sample information is stored at the start of a MOD file, and contains all the
relevant information for each of the 31 samples. This includes its name,
length, loop points, finetune etc..
So from here we loop 31 times and read in a block of information about the
sample according to the loop counter.
PseudoCode:
===========
- from this point, loop 31 times
- for the sample # <loopcounter>....
- read in 22 bytes, store as SAMPLE_NAME
- read in 2 bytes (word), store as SAMPLE_LENGTH * \
- read in 1 byte, store as FINE_TUNE @ /\ IMPORTANT:
- read in 1 byte, store as VOLUME > see key
- read in 2 bytes (word), store as LOOP_START * \/ below
- read in 2 bytes (word), store as LOOP_LENGTH * /
- end of loop
KEY:
====
* To get the real value in bytes, calculate it with (byte1*100h + byte2) * 2
@ For FINE_TUNE, if the value is > 7, subtract 16 from it to get the signed
value (i.e. 0-7 = 0-7, and 8-15 = -8 to -1)
Storage Issue:
==============
I think the best way to store information on the 31 instruments, is to store
its information in a structure, then have an array of 31 of these instrument
structures. Like this:
struct SAMPLE {
char SAMPLE_NAME[22]
word SAMPLE_LENGTH
byte FINE_TUNE
byte VOLUME
word LOOP_START
word LOOP_LENGTH
(also some physical position information - see sample loading section.
some possibilities are under GUS...
dword GUS_OFFSET (offset in gus dram)
OR using main memory with sb say..
char *SAMP_BUFF (pointer to the actual physical data in memory)
}
now declare an array of 31 SAMPLEs. I do this in the general mod header
structure which is explained fully in the next section.
The other way which can be used is just to keep a heap of global arrays like
this;
char SAMPLE_NAME[31][22]
word SAMPLE_LENGTH[31]
byte FINE_TUNE[31]
byte VOLUME[31]
word LOOP_START[31]
word LOOP_LENGTH[31]
Suggestion:
===========
Now code a little viewer once you have done this to make sure everything is
stored properly. This is VERY a important step. Compare your output to
the tracker it came from or a player that shows all sample information.
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² 2.5 Load Order Information ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
Explanation:
============
Ok now sample information is loaded, the next section of the module contains
order information. Order information in a mod defines in what order patterns
are going to be played. This means the composer could set orders 0 and 1
to pattern 0, for example, and the intent would be for pattern 0 to play
twice. Its entry in the order table would look like this.
ORDER : 0 1 2 3 4 5 6 7 8 9
PATTERN: 0 0
Note orders have to be from 0 to length of song, but patterns can be chopped
and changed around in any order.
The first byte from here will tell us the length of the song in -orders-,
even though they are stored in 128 bytes of information.
PsuedoCode:
===========
- read a byte, store as SONG_LENGTH (this is the number of orders in a song)
- read a byte, discard it (this is the UNUSED byte - used to be used in PT as
the restart position, but it is usually 127 so it isn't viable at all..)
Now we are at the orders table, this is 128 bytes long and contains the order
of patterns that are to be played in the song. here we have to find out how
many physical patterns there are in the module. How do we do this? Simple
just check every order byte and the highest value found is stored as the
number of patterns in the song.
- set NUMBER_OF_PATTERNS to equal 0
- from this point, loop 128 times
- read 1 byte, store it as ORDER <loopcounter>
- if this value was bigger than NUMBER_OF_PATTERNS then set it to that
value.
- end of loop
- read 4 bytes, discard them (we are at position 1080 again, this is M.K. etc!)
Storage Issue:
==============
One way is to go back to the other original MODhead structure, which contained
general information about the mod. here is the entire structure.
struct MODHEADER {
char NAME[20] ; song name
SAMPLE INST[31] ; instrument headers
byte SONG_LENGTH ; song length
byte NUMBER_OF_PATTERNS ; number of physical patterns
byte ORDER[128] ; pattern playing orders
} MODHEAD;
or the second way would just to be store them all as global variables
char NAME[20] ; song name
byte SONG_LENGTH ; song length
byte NUMBER_OF_PATTERNS ; number of physical patterns
byte ORDER[128] ; pattern playing orders
no array of samples here because if you saw the sample header loading section
we just stored them all as their own arrays.
Suggestion:
===========
As always print out the 128 orders, and see if the pattern numbers displayed
are correct. Now you should have a viewer that can just about display every
bit of information about the module! OK that stuff was easy. Now it's time
for something tougher.. the pattern data!
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² 2.6 Load Pattern Data ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
Explanation:
============
This is about the hardest part to code of the LOADER, and storage issues here
are VERY important, so it will be discussed first. I'm going to try and be
as general as I can as I don't want to appear to be trying to steer you in
any direction, but I will be specific enough to guide you.
Storage Issues:
===============
There are only a few ways to store pattern data, I've spent some time
pondering this issue.
Patterns really need to be stored DYNAMICALLY, or in other words only use as
much memory as you need.
So using a fixed array is not a good idea. The best way is using pointers,
one to point to the actual pattern data, and another to rove through the
pattern data while playing/loading and pluck out the right notes.
I found the most viable methods of storing pattern data are -
1. Creating a pointer to ALL pattern data. i.e. one big chunk.
Most players use this method, including the GUSplay routines
by Cascada and even Protracker by Lars Hamre. Older versions of FMOD used
this method also.
Diagramatically it looks like this:
Offset 0 x 2x 3x 4x 5x 6x 7x 8x 9x
ÚÄÄÄÄÄÄÂÄÄÄÄÄÄÂÄÄÄÄÄÄÂÄÄÄÄÄÄÂÄÄÄÄÄÄÂÄÄÄÄÄÄÂÄÄÄÄÄÄÂÄÄÄÄÄÄÂÄÄÄÄÄÄ¿
PatternBuff Ä´ PAT0 PAT1 PAT2 PAT3 PAT4 PAT5 PAT6 PAT7 PAT8 ³
ÀÄÄÄÄÄÄÁÄÄÄÄÄÄÁÄÄÄÄÄÄÁÄÄÄÄÄÄÁÄÄÄÄÄÄÁÄÄÄÄÄÄÁÄÄÄÄÄÄÁÄÄÄÄÄÄÁÄÄÄÄÄÄÙ
The Problem with this is in REAL MODE DOS we have 64kb segments, which can
cause some problems as you have to change segments to access the next 64kb
of memory. Segment/offset arithmetic needs to be used to avoid wraparound,
or you can let it be done for you, as is shown next.
Ways to avoid this segment wraparound are:
1. Use protected mode for flat access to all available memory - no wrap.
The normal ways to do this are to use DOS4G/W (tm) Rational Systems,
or PMODE/W by Charles Scheffold and Thomas Pytel. If you are a C/C++
programmer you can only use Watcom C++ to use these systems.
Other protected mode handlers are Borland Powerpack (which is shit btw)
or you can write your own DPMI routines.
2. In (borland) C, use HUGE pointers. i.e. char huge *buffer; This lets
you access all of the 640kb of base memory as usual, but it does the
segment/offset calculation for you! and AUTOMATICALLY. It is a great
feature of C, but is also slower than using protected mode because it
has to do the calculations every time you reference it.
You use farmallocs etc to allocate the memory. Look up more in the
manuals for more information.
2. Create an array of pointers, i.e. a seperate pointer for each pattern.
This method has to be used with new formats supporting variable lengthed
patterns. It has an advantage for real mode, but otherwise for a .MOD/S3M
player which are 64 rows long, method 1 is just as good.
Diagramatically it looks like this:
ÚÄÄÄÄÄÄ¿
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ´ PAT0 ³
³ ÚÄÄÄÄÄÄ¿ ÀÄÄÄÄÄÄÙ ÚÄÄÄÄÄÄ¿
³ ÚÄÄÄ´ PAT1 ³ ÚÄÄÄÄÄÄ¿ ³ PAT3 ³
PatternBuff[0] ÄÄÙ ³ ÀÄÄÄÄÄÄÙ ³ PAT2 ³ ÀÄÄÂÄÄÄÙ
PatternBuff[1] ÄÄÄÄÙ ÚÄÄÄÄÄÄÄÁÄÄÄÄÄÄÙ ³ ÚÄÄÄÄÄÄ¿
PatternBuff[2] ÄÄÄÄÄÄÄÄÄÙ ÚÄÄÄÄÄÄÄÄÄÄÄÙ ÚÄ´ PAT4 ³
PatternBuff[3] ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ ÚÄÄÄÄÄÄÄÙ ÀÄÄÄÄÄÄÙ
PatternBuff[4] ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ ÚÄÄÄÄÄÄ¿
PatternBuff[5] ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ´ PAT5 ³ ÚÄÄÄÄÄÄ¿
PatternBuff[6] ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿ ÀÄÄÄÄÄÄÙ ³ PAT6 ³
PatternBuff[7] ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿ ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÁÄÄÄÄÄÄÙ
PatternBuff[8] ÄÄÄÄÄÄÄÄ¿ ÚÄÄÁÄÄÄ¿
PatternBuff[9] Ä ÚÄÄÁÄÄÄ¿ ³ PAT7 ³
PatternBuff[10] Ä ³ PAT8 ³ ÀÄÄÄÄÄÄÙ
PatternBuff[11] Ä ÀÄÄÄÄÄÄÙ
PatternBuff[12] Ä
...
PatternBuff[255] Ä
Say you were doing this in C you would declare it like this:
char far *patbuff[256]
This creates 256 pointers. One for each pattern as you like.
In the above diagram with 0-8 patterns, the other pointers 9-255 are
not allocated any memory so just dangle around in cyberspace. They are
not of concern. They don't cost any memory and if we don't touch them
then we wont get any problems.
3. You CAN avoid having this array of 256 pointers, and make it more dynamic
by using a DOUBLE POINTER.
A double pointer is basically a pointer to a list of pointers..
got that? :)
say in C, the declaration would look like this.
char **patbuff;
first you allocate memory for the number of patterns you think you will
use.. ie.
patbuff = (char *)malloc(NUMBER_OF_PATTERNS * sizeof(char *));
Then to allocate memory for each pattern individually, do it like you
would for the array method discussed in method 2.
Advantages of Method 2 and 3 over method 1:
===========================================
- You don't have to worry about wraparound any more, or going over segment
boundaries. (because at most a pattern could be is 8192 bytes! - and
that's a 32 channel pattern with 64 rows and 4 bytes per note)
- Only far pointers are needed now, not huge pointers. Saves calculation.
- Instead of calculating the pattern offset value whilst playing, you just
reference the correct pattern pointer using the pattern number as the
index.
- ESSENTIAL for variable lengthed patterns in formats like XM.
- Easier to use these small pages for EMS memory handling.
Disadvantages:
==============
- Fragmentation (not really a problem)
When allocating memory, you need to know how much memory you need to use.
Using the one huge pattern buffer method, It works this way:
- declare a pointer and allocate it the amount of memory calculated below;
CHANNELS * 4 * 64 * (NUMBER_OF_PATTERNS+1)
³ ³
³ ÀÄÄÄ (rows per channel)
ÀÄÄÄÄÄÄÄ (bytes per note)
Why add 1 to NUMBER_OF_PATTERNS? well because patterns start at 0, and finish
at NUMBER_OF_PATTERNS, hence the additional 1. If you didn't add 1 and there
was only 1 pattern you would end up allocating 0 bytes for the pattern data.
This value is normally going to be a very big number, so a dword will be
needed to store it.
So to find the physical pattern in your pattern buffer, calculate the offset
with the formula (CHANNELS * 4 * 64) * PATTERN_NUMBER.
Say we want to point to the start of pattern 4 in an 8CHN mod.
(8 * 4 * 64) * 4
= 8192.
So as you travel through this pattern just increment your pointer by 4 bytes
at a time.
A note is stored in the actual file as 4 bytes, it is done in this fashion.
The pseudocode below shows how to unravel this amigafied mess.
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ Byte 0 Byte 1 Byte 2 Byte 3 ³
³ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄij
³aaaaBBBB CCCCCCCCC DDDDeeee FFFFFFFFF³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
aaaaDDDD = sample number
BBBBCCCCCCCC = sample period value
eeee = effect number
FFFFFFFF = effect parameters
PseudoCode:
===========
- calculate amount of memory needed for NUMBER_OF_PATTERNS patterns like so:
CHANNELS * 4 * 64 * (NUMBER_OF_PATTERNS+1)
- create a base pointer and allocate the memory needed
- From this point, loop for as many times as NUMBER_OF_PATTERNS
- From this point, loop 64 * CHANNELS times (this equals 1 pattern)
- read 4 bytes
- store SAMPLE_NUMBER as (byte0 AND 0F0h) + (byte2 SHR 4)
- store PERIOD_FREQUENCY as ((byte0 AND 0Fh) SHL 8) + byte1;
- store EFFECT_NUMBER as byte2 AND 0Fh
- store EFFECT_PARAMETER as byte 3
- increment pattern pointer by 4 bytes
- end loop
- end loop
OK:
===
Alright so lets look at this again in simpler terms:
- We have a big buffer that is meant to store all the pattern data
- Then we start loading in the notes *4* bytes at a time, and unravel them
into something meaningful as shown above.
- store the new note variables one after the other, and it should fill the
buffer to the exact size as was allocated in the beginning.
Suggestion:
===========
With EFFECT_PARAMTER, you might be tempted to store the 2 values stored in
here as 2 seperate variables, e.g. EFFECT_PARAMETER_X, and EFFECT_PARAMETER_Y.
I used to store them this way but I assure you when you get into coding your
effects this method is quite inefficient, I saved memory and increased
speed (but not noticeably) just by storing them in the 1 byte, and splitting
them only in the few times that you do need it. (i.e, printing them out
separately, or vibrato, or for finding out which E (extra) effect to use etc.)
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² 2.6.1 Four bytes? ²±° ³ *IMPORTANT*
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
At this stage you're probably thinking.. how do I fit all this into only 4
bytes?
For a start, DON'T store the amiga periods as your note value. Convert each
period to a note number. **SEE SECTION 3.5.1** for more discussion on this,
and notes and frequencies. In summary you just scan through the amiga table
until it matches the value you loaded in, and give it a note value according
to a counter variable.
Anyway even if you did store the amiga period value as your note (which you
wont), then you can still fit it all into 4 bytes. The file did it so why
can't you?
I used bit allocation. This means I only use the bits I need in a byte, and
not a whole byte.
An example of this is the note volume is only capable of getting up to 64, so
we only need 6 bits. The sample number goes up to 31. This only needs 5
bits. Follow here and see how things are allocated. This is similar to the
way I do it in my player. In C you can allocate a variable and tell how many
bits you want to use per variable (make sure they add up to a multiple of 8!)
ASM:
====
In asm you would have to use a 4 byte note buffer, and do the bit
calculations yourself runtime, before you access them, which isn't too hard.
You could use a dword to store the whole note information, read it into
EAX say, then to pick out the effect parameter, which you stored in the lower
8 bits for example, all you have to do is reference AL!
Say you have your pattern data pointed to by DS:(E)SI. This is a rough
example of how to select a note and pick out the relevant information.
mov eax, dword ptr ds:[si] ; eax now holds the whole note's info
; (or even just use a simple LODSD)
mov ebx, eax
shr ebx, 21
mov note, ebx ; note number (11 bits)
mov ebx, eax
shr ebx, 16
and bx, 31
mov sample, bx ; sample number (5 bits)
mov effectno, ah ; effect number (8 bits)
mov effectparm, al ; effect parameter (8 bits)
to move to the next note:
add si, 4
(this isn't necessary if you used LODSD as it increments SI for you - although
remember that a single lodsd is a cycle or 2 slower than using a mov as well
as an add)
In this case it would probably be better to store the note and sample number
in the lower 16 bits instead of the effect information, because note and
number are accessed a lot more.
C++ example:
============
int note:11; // 0-?? = 11 bits = 0-2048 should be plenty for your needs.
byte number:5; // 0-31 = 5 bits
byte effect; // 0-15 = 4 bits, but use 8 to keep things even
byte eparm; // 0-255 = 8 bits
Even smaller!:
==============
For one version of FMOD (not released), I actually used 3 bytes!!
For this I used 7 bits to store the note, which has a range of 0-127.
I did this by not counting fine tunes in my note value (i.e. notes were
stored as 1 note apart, not 8 finetunes apart like normal, so it requires 8
times less space).
So for finetuning I converted finetunes to a middle C value in hz like S3M
(see FS3MDOC.TXT on how this works), therefore I only needed the amiga table
for the actual notes and not the finetunes between. (the tuning is worked
out mathematically runtime.)
So I got something like:
note = 7bits
number = 5bits
effect = 4bits
eparam = 8bits
----------------------------
= 24bits = 3bytes!
I ditched it when I supported S3M because it needs room for 99 instruments,
more possible notes (because of 9 octaves), has a volume byte,
and more effects.
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² 2.7 Load Sample Data ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
PsuedoCode:
===========
- From this point, loop 31 times
- get the length of the sample # <loopcounter> (stored in your variable)
At this point I use only GUS, and dump the sample to the GUS dram, but if you
were using Sound Blaster etc, you would just declare 31 pointers in memory
and allocate them a SAMPLE_LENGTH sized buffer, then load the information
into those buffers. When you need to play them you would mix
the channels into a small buffer then DMA that buffer out to the sound card.
See FS3MDOC.TXT for more about mixing and Sound Blaster.
- [SOUNDBLASTER] allocate a SAMPLE_LENGTH sized pointer to buffer in
memory and load the sample into it
- [DRAM-BASED-CARD (GUS)] poke/DMA bytes into DRAM, increment dword
offset value GUS_OFFSET, and store that value next to the sample's
information (along side length, volume, finetune, loop start etc)
- check that your samples fit into (D)RAM, and exit with an error if
they don't.
- end loop
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² 2.8 Phew :) ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
Wasn't that bad was it? Now you have the FULL mod file stored away at your
disposal, with samples ready to blast.
Suggestions:
============
Now is a GOOD time to do some thorough testing. Do these things
- Make sure your sample headers and information are stored correctly
- Make sure your pattern data is stored perfectly.. it's quite important you
know..
- Make sure your samples are stable in memory, and try to play them through
your sound card.. you can have a few problems with misloaded samples I have
found.. Also make sure the loop points are played correctly!
- Make sure you deallocate your memory before quitting the program!!
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² : SECTION 3 : ²±° ³
³ °±² Playing the MOD ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² 3.1 OK where do I start ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
I think the main thing you need to do now once you are satisfied your MOD is
loaded properly, is to set up an interrupt function, and understand a bit
about the way a MOD is played.
I'm going to use the system timer to hook onto here as an example, and if you
want to use other interrupt servicers you can do that if you know how..
(i.e. GUS IRQ).
You should know how to set up an interrupt handler yourself, but ill describe
how to do it here with a bit of code to demonstrate.
The system timer lies on INT 8
- Get the old handlers vector for int 8h, and store it away for later
- Set your new handler function to the vector for int 8h
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ REMEMBER TO REHOOK YOUR OLD TIMER TO ITS ORIGINAL PLACE WHEN THE SONG IS ³
³ FINISHED! ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
In C you would do that like this:
=================================
oldhandler = getvect(8);
setvect(8, handler);
- where oldhandler has to have the prototype globally declared as
void interrupt ( *oldhandler)(...); // (...) for C++, () for C.
- for dummies the actual handler function looks like this
void interrupt modhandler(...) {
... // do main loop here
oldmodhandler(); // this is here to return int8 to what it
// normally did. It'll crash without it.
}
In PASCAL it would look something like this:
============================================
GetIntVec($8, Addr(OldTimer));
SetIntVec($8, Addr(ModInterrupt));
- with the function looking something like this
(I have no idea if this is right as I don't do pascal)
{ $ F+,S-,W-}
Procedure modhandler; Interrupt;
Begin
...
OldTimer;
End;
{ $ F-,S+}
If you're still not sure in C or pascal, check out the online manual on
getvect/setvect etc.
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² 3.1.1 Speed vs BPM ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
Some people have asked me to explain more about this because they just don't
seem to get it at first. They are DIFFERENT and you need a global SPEED
variable and a BPM variable to store them.
SPEED is the number of ticks (or times your interrupt handler is called)
between each time you process a new row.
In a module, say you have a speed of 6. This means the interrupt handler
should tick 6 times for every row of the pattern processed. Now imagine
if the speed was 3, it would take 1/2 the time to process the rows,
therefore making the song play twice as fast.
NOTE: The default speed to set a song to before it is played is 6. The
composer might not set a speed and just use that default speed.
BPM, also called tempo, is the rate at which your interrupt handler should
be running. It is the hardware rate in HZ that your timer should be running
at, and the next section, SECTION 3.2 shows you how to set the PC's system
timer.
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² 3.2 Setting the timer's speed ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
Ok now your interrupt handler is already firing.. so one thing you must
do is set it to the right rate, we don't want mods that play way to fast or
slow, we want it at 125 BPM right now (or 50hz, or 50 ticks a second - the
reason it is 50 is thanks to the formula HZ = 2*bpm/5, check out more on this
below).
How do you set the system timer's speed? if we want 50hz, we have to use
a divisor to calculate the right rate like so.
Hz = 50
PCTimerRate = 1193180/Hz ;<- 1193180 is the divisor using Hz as the rate
Then use the below code to set the PC timer's rate to the desired speed.
Below tells us how to convert BPM to HZ.
ASSEMBLER:
==========
mov dx, 043h
mov al, 036h
out dx, al
mov dx, 040h
mov ax, PCTimerRate ; here's the PCTimerRate variable
out dx, al ; write the lower 8 bits of the value
shr ax, 8
out dx, al ; now write the upper 8 bits of the value
C/C++:
======
outportb(0x43, 0x36);
outportb(0x40, PCTimerRate & 0xFF); // lower 8 bits
outportb(0x40, PCTimerRate >> 8); // upper 8 bits
PASCAL:
=======
Port [$43] := $36;
Port [$40] := (PCTimerRate AND $FF);
Port [$40] := (PCTimerRate SHR 8);
Now the interrupt function should be ticking away at 50 times a second.
For other BPM's, which will be used because of the change tempo effect Fxy
where xy is 20h and up, we need to convert BPM to HZ. If the Fxy effect
is set to below 20h, then you change the SPEED and not the BPM. This is
looked at later on. (See section 5.15)
To convert BPM to HZ, you use :
HZ = 2 * BPM / 5 (i.e 125bpm = 50hz)
then PC_TIMER_RATE = 1193180 / HZ, for the set timer routine.
Simple huh. You'll need this for effect Fxy, but don't worry about this
until later. For now just get it working at 125bpm. (50hz)
Effect Fxy is explained in SECTION 5.15
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² 3.3 Player Logic ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
Now lets take a look at the interrupt function, this is where the playing
is done.
The SPEED of a song is the base on how your mod is played. Each row of a
pattern is updated every SPEED number of clock ticks, so if a speed is 6,
then you only update each row every 6 clock ticks. So on a speed like 3,
the row is going to be updated every 3 ticks and will play twice as fast as
speed 6.
Inbetween you update certain tick sensitive effects, like portamentos,
volume slides and vibrato.
Diagramatically the playing of a mod looks like this.
SPEED IS 6
tick#
ÚÄÙ
0: UPDATE ROW #0 <- update the 4,6 or 8 notes here in a mod's row.
1: --- \
2: --- \
3: --- >- certain effects are updated here
4: --- /
5: --- /
0: UPDATE ROW #1
1: ---
2: ---
3: ---
4: ---
5: ---
0: UPDATE ROW #2
etc..
Logically a very basic representation of playing a mod looks like this:
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ STATIC TICK = SPEED ³ - declaration, start it off at SPEED, not 0, as we
³ ³ want straight into the 'if tick >= speed condition'
³ TICK = TICK + 1 ³ - now increment the tick counter
³ if TICK >= SPEED ³ - if the tick # is bigger or equal than SPEED then
³ update_row ³ - update the CHANNEL number of notes for the new row
³ tick =0 ³ - reset tick to 0
³ ROW = ROW + 1 ³ - increment our row
³ else update_effect ³ - else we update the tick based effects.
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
But you will have to take into account there are only 64 rows in a pattern,
and if you hit 64 then jump to the next pattern and start at row 0 again.
I say 64 because row 63's effects have to be played out before you jump to
the next pattern.
don't bother with update_effect for some time until you have got update_row
going OK.
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
³ °±² 3.3.1 Orders/Patterns ²±° ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
Just a short note on this.
When you reach the end of the pattern or whatever, you need to go to the next
order. Now say you had your pattern numbers stored in an order array as they
should be, then it is simply a task of referencing that pattern number
according to the index ORDER, and then repositioning your pattern pointer
accordingly.
i.e. If your order list is something like this.
ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
Order ³ 0 1 2 3 4 5 6 7 8 9 .... ³
³ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄij
Pattern ³ 0 0 1 4 5 2 3 4 4 6 .... ³