summaryrefslogtreecommitdiff
path: root/plugins/dumb/dumb-kode54/docs/dumbfull.txt
blob: 6a8e192c310c8e3a916f9b907e3a7f66037db3d5 (plain)
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
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
1025
1026
1027
1028
1029
1030
1031
1032
1033
1034
1035
1036
1037
1038
1039
1040
1041
1042
1043
1044
1045
1046
1047
1048
1049
1050
1051
1052
1053
1054
1055
1056
1057
1058
1059
1060
1061
1062
1063
1064
1065
1066
1067
1068
1069
1070
1071
1072
1073
1074
1075
1076
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
1090
1091
1092
1093
1094
1095
1096
1097
1098
1099
1100
1101
1102
1103
1104
1105
1106
1107
1108
1109
1110
1111
1112
1113
1114
1115
1116
1117
1118
1119
1120
1121
1122
1123
1124
1125
1126
1127
1128
1129
1130
1131
1132
1133
1134
1135
1136
1137
1138
1139
1140
1141
1142
1143
1144
1145
1146
1147
1148
1149
1150
1151
1152
1153
1154
1155
1156
1157
1158
1159
1160
1161
1162
1163
1164
1165
1166
1167
1168
1169
1170
1171
1172
1173
1174
1175
1176
1177
1178
1179
1180
1181
1182
1183
1184
1185
1186
1187
1188
1189
1190
1191
1192
1193
1194
1195
1196
1197
1198
1199
1200
1201
1202
1203
1204
1205
1206
1207
1208
1209
1210
1211
1212
1213
1214
1215
1216
1217
1218
1219
1220
1221
1222
1223
1224
1225
1226
1227
1228
1229
1230
1231
1232
1233
1234
1235
1236
1237
1238
1239
1240
1241
1242
1243
1244
1245
1246
1247
1248
1249
1250
1251
1252
1253
1254
1255
1256
1257
1258
1259
1260
1261
1262
1263
1264
1265
1266
1267
1268
1269
1270
1271
1272
1273
1274
1275
1276
1277
1278
1279
1280
1281
1282
1283
1284
1285
1286
1287
1288
1289
1290
1291
1292
1293
1294
1295
1296
1297
1298
1299
1300
1301
1302
1303
1304
1305
1306
1307
1308
1309
1310
1311
1312
1313
1314
1315
1316
1317
1318
1319
1320
1321
1322
1323
1324
1325
1326
1327
1328
1329
1330
1331
1332
1333
1334
1335
1336
1337
1338
1339
1340
1341
1342
1343
1344
1345
1346
1347
1348
1349
1350
1351
1352
1353
1354
1355
1356
1357
1358
1359
1360
1361
1362
1363
1364
1365
1366
1367
1368
1369
1370
1371
1372
1373
1374
1375
1376
1377
1378
1379
1380
1381
1382
1383
1384
1385
1386
1387
1388
1389
1390
1391
1392
1393
1394
1395
1396
1397
1398
1399
1400
1401
1402
1403
1404
1405
1406
1407
1408
1409
1410
1411
1412
1413
1414
1415
1416
1417
1418
1419
1420
1421
1422
1423
1424
1425
1426
1427
1428
1429
1430
1431
1432
1433
1434
1435
1436
1437
1438
1439
1440
1441
1442
1443
1444
1445
1446
1447
1448
1449
1450
1451
1452
1453
1454
1455
1456
1457
1458
1459
1460
1461
1462
1463
1464
1465
1466
1467
1468
1469
1470
1471
1472
1473
1474
1475
1476
1477
1478
1479
1480
1481
1482
1483
1484
1485
1486
1487
1488
1489
1490
1491
1492
1493
1494
1495
1496
1497
1498
1499
1500
1501
1502
1503
1504
1505
1506
1507
1508
1509
1510
1511
1512
1513
1514
1515
1516
1517
1518
1519
1520
1521
1522
1523
1524
1525
1526
1527
1528
1529
1530
1531
1532
1533
1534
1535
1536
1537
1538
1539
1540
1541
1542
1543
1544
1545
1546
1547
1548
1549
1550
1551
1552
1553
1554
1555
1556
1557
1558
1559
1560
1561
1562
1563
1564
1565
1566
1567
1568
1569
1570
1571
1572
1573
1574
1575
1576
1577
1578
1579
1580
1581
1582
1583
1584
1585
1586
1587
1588
1589
1590
1591
1592
1593
1594
1595
1596
1597
1598
1599
1600
1601
1602
1603
1604
1605
1606
1607
1608
1609
1610
1611
1612
1613
1614
1615
1616
1617
1618
1619
1620
1621
1622
1623
1624
1625
1626
1627
1628
1629
1630
1631
1632
1633
1634
1635
1636
1637
1638
1639
1640
1641
1642
1643
1644
1645
1646
1647
1648
1649
1650
1651
1652
1653
1654
1655
1656
1657
1658
1659
1660
1661
1662
1663
1664
1665
1666
1667
1668
1669
1670
1671
1672
1673
1674
1675
1676
1677
1678
1679
1680
1681
1682
1683
1684
1685
1686
1687
1688
1689
1690
1691
1692
1693
1694
1695
1696
1697
1698
1699
1700
1701
1702
1703
1704
1705
1706
1707
1708
1709
1710
1711
1712
1713
1714
1715
1716
1717
/*  _______         ____    __         ___    ___
 * \    _  \       \    /  \  /       \   \  /   /       '   '  '
 *  |  | \  \       |  |    ||         |   \/   |         .      .
 *  |  |  |  |      |  |    ||         ||\  /|  |
 *  |  |  |  |      |  |    ||         || \/ |  |         '  '  '
 *  |  |  |  |      |  |    ||         ||    |  |         .      .
 *  |  |_/  /        \  \__//          ||    |  |
 * /_______/ynamic    \____/niversal  /__\  /____\usic   /|  .  . ibliotheque
 *                                                      /  \
 *                                                     / .  \
 * dumb.txt - DUMB library reference.                 / / \  \
 *                                                   | <  /   \_
 * See readme.txt for general information on         |  \/ /\   /
 * DUMB and how to set it up.                         \_  /  > /
 *                                                      | \ / /
 * If you are new to DUMB, see howto.txt.               |  ' /
 *                                                       \__/
 */


***********************************
*** Include Files and Libraries ***
***********************************


dumb.h

   Include this if you only want the core DUMB library functions. You will
   be able to load music files and render them into memory buffers at your
   own pace. The core library is completely portable, and as such does not
   access hardware; you must relay the sound data to the sound card yourself.
   A stdio file input module is available, but you must actively register it
   if you wish to use it (see dumb_register_stdfiles()); if you do not
   register it, it will not be linked into your executable. You must register
   it in order to load stand-alone music files.

   Optimised: libdumb.a  (-ldumb)
   Debugging: libdumbd.a (-ldumbd)


aldumb.h

   Include this if you wish to use DUMB with Allegro. This will provide you
   with functions to play DUHs back through Allegro's audio streams and embed
   music files in Allegro datafiles. A file input module using Allegro's
   packfiles is provided; you have a choice between this and the stdio
   module. You will be able to load datafiles containing music files no
   matter which file input module you register, or even if you register no
   file input module. However, you must register a file input module in order
   to load stand-alone files.

   Optimised: -laldmb -lalleg -ldumb
   Debugging: -laldmd -lalld -ldumbd

   libaldmb.a or libaldmd.a must be linked in first, so the symbols can be
   resolved when linking in the other two libraries.


***********************************
*** Library Clean-up Management ***
***********************************


int dumb_atexit(void (*proc)(void));

   Registers a function to be called at the end of your program. You can
   register multiple functions to be called, and the one you register last
   will be called first. If you try to register the same function twice, the
   second attempt will have no effect.

   See fnptr.txt for help with function pointers.

   You must call dumb_exit() before exiting your program for this to work
   properly. The library itself registers functions with dumb_atexit(), so it
   is important to call dumb_exit() even if you do not use dumb_atexit()
   yourself.

   This function will return zero on success. It will return zero when
   trying to install the same function twice. If it fails through lack of
   memory, it will return nonzero. Generally you can ignore the return code;
   in the worst case some memory will not be freed at the end. If it is
   crucial that your function be called (e.g. to shut down some hardware or
   save critical data), then you should call your function manually at the
   end of the program instead of registering it here - or use the stdlib
   function atexit(), guaranteed under ANSI C to succeed for at least 32
   functions.


void dumb_exit(void);

   You should call this before exiting your program if you have used any part
   of DUMB in the program. Some parts of DUMB will allocate memory, and this
   function will free it all up.

   More specifically, this function will call any functions that have been
   registered with dumb_atexit(). If a part of DUMB needs shutting down, the
   shutdown procedure will have been registered in this way.

   dumb_exit() will, of course, also call any functions you registered with
   dumb_atexit() yourself.

   After a call to dumb_exit(), the list of functions is erased. If you are
   not ready to exit your program, you can start using DUMB anew as if your
   program had just started. (Note that not everything will be reset in
   practice - dumb_resampling_quality will retain whatever you set it to, for
   example, though you should not assume it will.)

   If you only need to call dumb_exit() once at the end of the program, you
   can use the following to register dumb_exit() with stdlib.h atexit():

      #include <stdlib.h>

      atexit(&dumb_exit);

   Then dumb_exit() will be called for you when your program exits. This is
   the recommended method, since it will ensure clean-up even if your program
   aborts. You should only call dumb_exit() manually if you need to shut DUMB
   down prematurely, or if atexit() is unavailable for one reason or another.


*****************************
*** Sequential File Input ***
*****************************


   DUMB provides a strictly sequential file input system which uses the
   DUMBFILE struct. "Strictly sequential" means you cannot seek backwards.
   However, the system will keep track of how many bytes you have read,
   enabling you to seek forwards. DUMBFILEs provide a convenient error
   detection system, so you do not have to check the return value from every
   function call in the way you do with the ANSI C functions.

   Note that DUMBFILEs cannot be used for output, nor can they be used
   portably for text files.

   If an error occurs when reading data from a DUMBFILE, the DUMBFILE will
   become inoperative. All subsequent activities on the DUMBFILE will return
   error codes without attempting to read from the file. The position in the
   file will also be forgotten. You can find out if this has happened at any
   stage with the dumbfile_error() function. You are still required to close
   the DUMBFILE, and the return value from dumbfile_close() will tell you if
   an error has occurred.

   This system allows you to input large chunks of your file, neither
   checking every return value nor wasting time accessing a file that has
   already experienced an error. However, before you allocate an amount of
   memory or read in a quantity of data depending on previous input from the
   file, you should always check that such input was valid. In particular you
   should passing zero or negative numbers to malloc() or dumbfile_getnc().

   DUMBFILEs can be hooked. In other words, you can specify your own
   functions to do the work of reading from a file. While DUMB contains two
   modules for this purpose, it does not set them up for you automatically.
   In most cases you must register one of these modules yourself, or provide
   your own module. See register_dumbfile_system(), dumb_register_stdfiles()
   and dumb_register_packfiles().


void register_dumbfile_system(DUMBFILE_SYSTEM *dfs);

   Use this function to register a set of functions for use by the DUMBFILEs
   (a DUMBFILE system). The DUMBFILE_SYSTEM struct contains the following
   fields:

      void *(*open)(const char *filename);
      int (*skip)(void *f, long n);
      int (*getc)(void *f);
      long (*getnc)(char *ptr, long n, void *f);
      void (*close)(void *f);

   See fnptr.txt for help with function pointers such as these.

   Your 'open' function should open the file specified and return a pointer
   to a struct representing the open file. This pointer will be passed to
   your other functions as 'f'. Your 'close' function should close the file
   and free all memory pointed to by 'f'. Note that the 'close' operation
   should never be able to fail; if you are calling a function with a return
   value, you can generally ignore it.

   Your 'getc' function should read one byte from the file and return its
   value in the range 0 to 255. If an error occurs, you should return -1. Do
   not worry about remembering that an error has occurred; DUMB will do that
   for you.

   'skip' is for skipping parts of the file, and should skip n bytes,
   returning 0 on success or any other number on failure. 'getnc' should read
   n bytes from the file, store them at 'ptr', and return the number of bytes
   read (n on success, fewer on failure). However, these two functions are
   optional, and you should only provide them if the operations can be done
   more efficiently than with repeated calls to your 'getc' function. If this
   is not the case, specify NULL for 'skip', 'getnc' or both, and DUMB will
   use your 'getc' function to do the work.

   Once you have written all your functions, you need to create a
   DUMBFILE_SYSTEM struct to hold them, and pass its pointer to
   register_dumbfile_system().

   The DUMBFILE_SYSTEM struct must be permanent. In other words, it must be
   either global or static, and you should not modify it later. DUMB will not
   make its own copy.

   You will most likely create your own struct to represent the open file,
   but do not be tempted to specify that struct in the function prototypes
   and pacify the compiler warnings by casting your function pointers. There
   exist computer systems where a (void *) pointer and a (MY_STRUCT *)
   pointer are represented differently in memory, and a cast of such a
   pointer causes a tangible conversion to take place. If you cast the
   function pointers, the computer cannot know when such a conversion is
   necessary. Instead, use the following structure:

      int myskip(void *f, long n)
      {
         FILE *file = f;
         /* Do some stuff with 'file' */
         return something;
      }

   If you need examples, have a look at the two existing DUMBFILE systems in
   dumb/src/core/stdfile.c and dumb/src/allegro/packfile.c.


DUMBFILE *dumbfile_open(const char *filename);

   Open the specified file for input. You must pass the DUMBFILE pointer
   whenever you wish to operate on this file. When you have finished with the
   file, you must pass it to dumbfile_close().

   Before you use this function, make sure you have registered a DUMBFILE
   system. See register_dumbfile_system(), dumb_register_stdfiles() and
   dumb_register_packfiles().


DUMBFILE *dumbfile_open_ex(void *file, DUMBFILE_SYSTEM *dfs);

   This function is provided for more specialised use. You should create a
   DUMBFILE_SYSTEM specially for the purpose. Its 'open' field is irrelevant;
   for neatness, set it to NULL, unless you are using this DUMBFILE_SYSTEM
   with register_dumbfile_system() as well.

   When you have called this function, the DUMBFILE struct it returned can be
   used as normal. The specified DUMBFILE_SYSTEM will be used for all input,
   with 'file' passed to your 'skip', 'getc' and 'getnc' functions as 'f'.
   This can be used, for example, to read from an already open file.

   Note that the position will always be initialised to 0 for this DUMBFILE.
   This means for example that offsets in the file do not need adjusting when
   embedding data in a larger file.

   There are two ways to use this function. If you want 'file' to persist
   after using a DUMBFILE returned by this function, you should make sure the
   'close' field in the DUMBFILE is set to NULL. When the DUMBFILE is closed,
   'file' will be left alone, and you can and should deal with it yourself
   when the DUMBFILE has been closed.

   Alternatively, you can provide a 'close' function to get rid of 'file' for
   you when the DUMBFILE is closed. If you do this, you should not otherwise
   use 'file' after a call to this function.

   If dumbfile_open_ex() has to return NULL, owing to lack of memory, then
   your 'close' function will be called if provided. In other words, if you
   have provided a 'close' function, then you no longer need to worry about
   'file' whether this function succeeds or not.

   See dumb/src/helpers/stdfile.c and dumb/src/allegro/packfile.c for
   examples of how to use this function. Neither provides a 'close' function,
   so I hope my explanation here will suffice. If not, please feel free to
   contact me so I can make the explanation clearer and help you do what you
   want to do. Contact details are at the end of this file.


long dumbfile_pos(DUMBFILE *f);

   Returns the number of bytes read from the DUMBFILE (or skipped) since it
   was opened, or -1 if an error has occurred while reading.


int dumbfile_skip(DUMBFILE *f, long n);

   Skips n bytes of the specified DUMBFILE. Returns zero on success.


int dumbfile_getc(DUMBFILE *f);

   Reads one byte from the DUMBFILE and returns it in unsigned format (from 0
   to 255). If an error occurs, or occurred before, this function returns -1.


int dumbfile_igetw(DUMBFILE *f);

   Reads two bytes from the DUMBFILE and combines them into a word ranging
   from 0 to 65535. The first byte read is the least significant byte, as
   with Intel processors. This function returns -1 on error.


int dumbfile_mgetw(DUMBFILE *f);

   Reads two bytes from the DUMBFILE and combines them into a word ranging
   from 0 to 65535. The first byte read is the most significant byte, as
   with the Apple Macintosh. This function returns -1 on error.


long dumbfile_igetl(DUMBFILE *f);

   Reads four bytes from the DUMBFILE and combines them into a long integer
   ranging from -2147483648 to 2147483647. The first byte read is the least
   significant byte, as with Intel processors. This function returns -1 on
   error, but -1 is also a valid return value. After a call to this function,
   you can use dumbfile_error() to find out if an error occurred.


long dumbfile_mgetl(DUMBFILE *f);

   Reads four bytes from the DUMBFILE and combines them into a long integer
   ranging from -2147483648 to 2147483647. The first byte read is the most
   significant byte, as with the Apple Macintosh. This function returns -1 on
   error, but -1 is also a valid return value. After a call to this function,
   you can use dumbfile_error() to find out if an error occurred.


unsigned long dumbfile_cgetul(DUMBFILE *f);

   Reads an unsigned (nonnegative) integer from the DUMBFILE. The integer is
   stored in a condensed format where smaller numbers use less space:

           0 to 127         1 byte
         128 to 16383       2 bytes
       16384 to 2097151     3 bytes
     2097152 to 268435455   4 bytes
   268435456 to 4294967295  5 bytes

   This format is the same as that used for the times between notes in MIDI
   files.

   If an error occurs, this function returns (unsigned long)(-1), but that
   may be a valid return value. After a call to this function, you can use
   dumbfile_error() to find out if an error occurred.


signed long dumbfile_cgetsl(DUMBFILE *f);

   Reads a signed integer from the DUMBFILE. The integer is stored in a
   condensed format where numbers closer to zero use less space:

           -64 to 63          1 byte
         -8192 to 8191        2 bytes
      -1048576 to 1048575     3 bytes
    -134217728 to 134217727   4 bytes
   -2147483648 to 2147483647  5 bytes

   If an error occurs, this function returns -1, but -1 is also a valid
   return value. After a call to this function, you can use dumbfile_error()
   to find out if an error occurred.


long dumbfile_getnc(char *ptr, long n, DUMBFILE *f);

   Reads n bytes from the DUMBFILE and stores them at 'ptr'. Note that the
   pointer is to a series of chars. You may also use this function to read in
   a series of signed chars or unsigned chars (which are both officially
   distinct types from char), but do not use this to read ints, structs or
   any other data type from the file. Integers must be read one at a time
   using dumbfile_igetl(), dumbfile_cgetul(), etc. To load a struct in, you
   must read each field separately using an appropriate function for each
   one. For complicated datatypes, you can simplify this process by writing a
   function for each struct.


int dumbfile_error(DUMBFILE *f);

   This function returns -1 if an error has occurred with the specified
   DUMBFILE, or 0 if all is well.


int dumbfile_close(DUMBFILE *f);

   This function closes the DUMBFILE, after which the pointer will be
   invalid. dumbfile_close() returns the value that dumbfile_error() would
   have returned, which is -1 if an error occurred while reading or 0
   otherwise. Regardless of the return value, the file will always be closed
   properly.


*******************************
*** stdio File Input Module ***
*******************************


void dumb_register_stdfiles(void);

   This function registers the stdio file input module for use by DUMBFILEs.
   FILE structs and their corresponding functions, as defined by the ANSI C
   header stdio.h, will be used internally for all DUMBFILE input (unless
   opened with dumbfile_open_ex()).

   This must be called before dumbfile_open() is used, or else an alternative
   system must be registered (see register_dumbfile_system() and
   dumb_register_packfiles()).


DUMBFILE *dumbfile_open_stdfile(FILE *p);

   If you have a stdio FILE struct representing an open file, you can call
   this if you wish to read from it using a DUMBFILE. This is useful when you
   need to pass a DUMBFILE struct to a library function, to read an embedded
   music file for example. When you close the DUMBFILE, you can continue
   using the FILE struct to read what follows the embedded data.


********************************
*** Signal Type Registration ***
********************************






   NOTE: SINCE THIS IS NOT TO BE THE INTRODUCTION TO DUMB, SHOULD THE
   FOLLOWING INFORMATION BE MOVED TO ANOTHER FILE?






   If you are lazy, then don't bother to read this section. Simply follow
   these steps:

      1. Call all the dumb_register_sigtype_*() functions. You must do this
         after dumb_init(), but before you try to call load_duh(). If you try
         to load any DUH files beforehand, the library will fail to load
         them.

      2. Run your program, and make sure the DUH file was successfully loaded
         and played.

      3. Comment out the first dumb_register_sigtype_*() function call.

      3.1. If the DUH file is still loaded and played, remove this function
           call.
      3.2. If the DUH file was not loaded and played, uncomment and keep this
           function call.

      4. Repeat Step 3 for the other function calls.

   Alternatively, the musician might have told you which of the functions you
   need to call.

   If you are the epitome of laziness, stop after Step 1. However, if you do
   this, your executable may contain unnecessary code.

   If, on the other hand, you are interested in how all this works, then read
   on.

   A DUH file comprises a number of 'signals'. A signal may be thought of as
   a 'black box'; commands go in, sound comes out. A signal may in turn call
   upon other signals to generate sound, and then do something with this
   sound to generate its own output.

   For example, here are the contents of a simple DUH file:
      Signal #0 SEQU
      Signal #1 SAMP
      Signal #2 SAMP
      Signal #3 SAMP

   'SEQU' and 'SAMP' are the signal types. 'SAMP' designates a sample, so
   Signals #1, #2 and #3 are simple recordings. For instance, Signal #1 could
   be a bass drum, Signal #2 a snare drum and Signal #3 a high hat. When
   invoked, these signals simply output their respective percussion sounds.

   'SEQU' designates a sequence. Signal #0 will therefore generate its sound
   by taking the output waveforms from the other signals and mixing them in
   at appropriate times. For example:

      0.00 Signal #1

      0.50 Signal #2

      1.00 Signal #1
      1.25           Signal #3
      1.50 Signal #2 Signal #3
      1.75           Signal #3
      2.00 Signal #1
      2.25           Signal #3
      2.50 Signal #2 Signal #3
      2.75           Signal #3
      3.00 Signal #1

      3.50 Signal #2 Signal #3

   The numbers down the left are times. Those with a good sense of rhythm
   will be able to see that this represents a rather lame, highly cheesy pop
   beat. Experienced gurus will also realise how painfully slow the beat is,
   given that the times are in seconds. But enough of that.

   A DUH is played by taking the output from Signal #0 and sending it through
   to the sound card via an Allegro audio stream. If you do not know what an
   audio stream is, suffice it to say that you must call Allegro's
   install_sound() function, and an audio stream uses one voice (hence one
   voice per DUH file). If you still don't know what I'm on about, read up on
   Allegro's digital sound system and then come back here. Alternatively,
   read porting.txt for details on how to use DUMB without Allegro.

   In reality, DUH files are likely to be much more complicated than the
   above example. Sequences may call upon other 'sub-sequences' to generate
   their sound, so the above beat might be used repeatedly by a
   'super-sequence' to generate a more complex (but still lame) piece of
   music.

   The DUH player library is split into two main parts - the core and the
   basic signal types. The core is very simple. It does not know what 'SAMP'
   or 'SEQU' mean. In fact it is not familiar with any signal type. It only
   contains generic code for dealing with signals and generating output. By
   itself, it cannot load or play any DUH file.

   In addition to the core, the player library comprises a few basic signal
   types, including 'SAMP' and 'SEQU'. In order to use a signal type, you
   must 'register' this signal type by calling its corresponding
   dumb_register_sigtype_*() function. These functions are listed below,
   along with descriptions of the signals.

   If you do not register a signal type, the code for that signal type will
   not be linked into your executable file. That means DUMB will not become
   bloated with age; new features will not add to the size of your executable
   unless you use them. Dynamically linked libraries will still increase in
   size.

   If you try to load a DUH file that uses a signal type you haven't
   registered, the library will not crash. It will simply fail to load the
   DUH file. In fact, an unrecognised or corrupt signal will prevent the
   library from reading the rest of the file - but remember, DUH files are
   always (intended to be) generated from other formats, so you will never
   lose any data this way.

   If you are unsure which signal types a DUH uses, get in contact with the
   author of the DUH file, or register them all and remove them one by one to
   find out which are required (as described in the steps for lazy people at
   the start of this section).

   The great advantage of DUMB over other music systems is that it enables
   you to create your own signals. Filters, synthesisers and compression
   algorithms can all be set up this way. Programming and audio experience
   are useful, but the editor provides a way to create your own signals even
   if you don't know C. More information is available in the on-line help in
   the editor.

   If you are interested in the gory details of how a signal works, or if you
   wish to brave creating your own without the help of the editor, then see
   the section entitled Signal Design.


void dumb_register_sigtype_sample(void);

   Registers the sample signal type ('SAMP'). This signal deals with samples,
   or recordings, in the DUH file. All samples are monaural; a combining
   signal can be used to combine two of these into a stereo sample. The DUMB
   library will cater for more than two channels to a certain extent, but it
   is currently limited by Allegro's audio streams.


void dumb_register_sigtype_combining(void);

   Registers the combining signal type ('COMB'). This signal takes a set of
   monaural signals and combines them into one signal with multiple channels.
   Stereo samples can be set up this way.


void dumb_register_sigtype_stereopan(void);

   Registers the stereo pan signal type ('SPAN'). This signal takes a
   monaural sample and sources it at the stereo position you specify,
   provided the DUH is played in stereo (or, more precisely, provided stereo
   output is requested of the signal).


void dumb_register_sigtype_sequence(void);

   Registers the sequence signal type ('SEQU'). This signal sequences the
   sound from other signals. It is what turns a random collection of
   recordings of single notes into a complete piece of music.


**********************
*** DUH Management ***
**********************


DUH *load_duh(const char *filename);

   Loads a .duh file. Before you call this, make sure you have registered all
   the necessary signal types (see 'Signal Registration'). The DUH is
   returned to you; you will need to pass it to various functions. When you
   have finished with it, call unload_duh() to remove it from memory.

   There is no special need to check that this function succeeds. All other
   functions can safely be called with a null pointer, which means your music
   will simply not play if it could not be loaded.


DUH *read_duh(DUMBFILE *f);

   Reads a DUH from an already open file, and leaves the file open so you can
   read subsequent data from the file if you wish. Otherwise this function is
   identical to load_duh().


void unload_duh(DUH *duh);

   Removes a DUH from memory. You must call this for all DUHs you load,
   making sure they're not playing at the time.


long duh_get_length(DUH *duh);

   Returns the length of a DUH; 65536 represents one second. This value is
   simply lifted from the DUH struct, so it may not truly correspond to the
   time for which the DUH will generate sound. However, if the musician is
   any good - or if the code that calculated this value is written properly -
   you can assume it represents the point at which the DUH first loops, or
   else it allows time for any final flourish to be appreciated. It is used
   by the Winamp plug-in to decide when to stop.


*******************************
*** Impulse Tracker Support ***
*******************************


int dumb_it_max_to_mix;

   Specifies the maximum number of samples DUMB will mix at any one time.
   The default number is 64. Regardless of this value, all samples will
   continue to be processed up to an internal maximum of 128 (slightly
   simplified), and cut samples will sound again as soon as the congestion
   clears. Samples are prioritised by final volume, after all factors
   affecting the volume of a sample have been considered.

   If you play two or more IT files at once, this value represents the
   maximum number of samples for each one. You will have to reduce it further
   if your computer cannot keep up.


DUH *dumb_load_it(const char *filename);

   Loads the specified Impulse Tracker file, encapsulating it in a DUH
   struct. No signal types need be registered for this to work. The length
   will be set to the point at which the music first loops (see
   duh_get_length()).

   Once the file is loaded, it can be treated exactly the same as any other
   DUH in memory.


DUH *dumb_read_it(DUMBFILE *f);

   Reads an Impulse Tracker file from an already open DUMBFILE. This leaves
   the DUMBFILE open, but the DUMBFILE may not be positioned at the end of
   the IT data. If you are embedding an IT in another file, you are advised
   to store the size of the IT file and make up for it at the end using
   dumbfile_pos().

   Otherwise, this function is identical to dumb_load_it().


*******************************
*** DUH Rendering Functions ***
*******************************


   Use these functions to generate samples from a DUH. First you call
   duh_start_renderer() with the DUH, the number of channels you want and the
   position at which you want to start. Then you use duh_render() to generate
   the samples. You can call duh_render() as many times as you like, and it
   will generate as many or as few samples as you require. When you have
   finished, call duh_end_renderer().


DUH_RENDERER *duh_start_renderer(DUH *duh, int n_channels, long pos);

   Starts a DUH_RENDERER off. This is the struct you can use to get samples
   from a DUH. This function does not generate any samples; you must pass the
   struct to duh_render() for that. When you have finished with it, you must
   pass it to duh_end_renderer(). You can use as many DUH_RENDERER structs as
   you like at the same time.

   Currently, n_channels can only be 1 or 2, for monaural and stereo sound
   respectively. The debugging library will cause your program to abort if
   you pass anything else. Future versions will be enhanced to support more
   channels as soon as someone needs them.

   When specifying the position, 0 represents the start of the DUH, and 65536
   represents one second. Unlike most other music systems, DUMB will always
   make sure every note is there right from the start (provided any custom
   signal types are properly designed). In other words, you can start a DUH
   at a point halfway through a long note, and you will still hear the long
   note.


long duh_render(
   DUH_RENDERER *dr,
   int bits, int unsign,
   float volume, float delta,
   long size, void *sptr
);

   Generates some samples. Pass the DUH_RENDERER as returned by
   duh_start_renderer(). Pass the number of bits, which should be 8 or 16. If
   unsign is nonzero, the samples will be unsigned (centred on 0x80 or 0x8000
   for 8 bits and 16 bits respectively). If unsign is zero, the samples will
   be signed.

   Allegro's audio streams always take unsigned samples. 8-bit .wav files
   always take unsigned samples. 16-bit .wav files always take signed
   samples.

   The volume is a float. 1.0f is the pseudo-maximum. If you pass 1.0f, any
   properly designed DUH will play nice and loud, but will not clip. You can
   pass a greater volume if you like, but be prepared for clipping to occur.
   Of course you can pass smaller values to play the DUH more quietly, and
   this will also resolve clipping issues in badly designed DUHs.

   Use delta to control the speed of the output signal. If you pass 1.0f, the
   resultant signal will be suitable for a 65536-Hz sampling rate (which
   isn't a commonly used rate). The most common sampling rates are 11025 Hz,
   22050 Hz, 44100 Hz and 48000 Hz. You can work out the required delta value
   as follows:

      delta = 65536.0f / sampling_rate;

   If you then increase this value, the DUH will speed up and increase in
   pitch. If you decrease it, the DUH will slow down and decrease in pitch.

   This function will attempt to render 'size' samples. In most cases it will
   succeed. However, if the end of the DUH is reached, it may render fewer.
   The number of samples rendered will be returned. Therefore, if the return
   value is less than the value of 'size' passed, you know the DUH has
   finished. It is safe to continue calling duh_render() if you wish, and it
   will continually return 0. However, if you wish to do this, you will
   probably have to fill the rest of the buffer with silence, which is 0 for
   signed, 0x80 for 8-bit unsigned or 0x8000 for 16-bit unsigned.

   The samples will be placed at sptr. Use an array of chars for 8 bits or an
   array of shorts for 16 bits. Stereo samples will be interleaved, left
   first. Your array should contain at least (size * n_channels) elements of
   the appropriate bit resolution.

   From an aesthetic standpoint if nothing else, it is wise to use the C
   qualifiers 'signed' or 'unsigned' depending on whether the samples are
   signed or unsigned. This is also convenient if you wish to process the
   samples further yourself.


long duh_renderer_get_position(DUH_RENDERER *dr);

   Tells you what position a DUH_RENDERER is up to, or -1 if it is invalid
   (perhaps owing to lack of memory). As usual, 65536 is one second. Note
   that this is a whole number, whereas a fractional part is stored
   internally; the sample will not be continuous if you terminate the
   DUH_RENDERER and then reinitiate it with the same position.


void duh_end_renderer(DUH_RENDERER *dr);

   Terminates a DUH_RENDERER. Be sure to call this when you've finished with
   one.


***********************************
*** Signal Design Helper Values ***
***********************************


DUMB_SEMITONE_BASE

   When a 'delta' value is required, you can use DUMB_SEMITONE_BASE to
   control the pitch. Use pow(DUMB_SEMITONE_BASE, n) to transpose up by n
   semitones. To transpose down, use negative n.


DUMB_QUARTERTONE_BASE

   When a 'delta' value is required, you can use DUMB_QUARTERTONE_BASE to
   control the pitch. Use pow(DUMB_QUARTERTONE_BASE, n) to transpose up by n
   quartertones. To transpose down, use negative n.


DUMB_PITCH_BASE

   When a 'delta' value is required, you can use DUMB_PITCH_BASE to control
   the pitch accurately. Use pow(DUMB_PITCH_BASE, n) to transpose up by n
   units, where 256 units represent one semitone. This scale is used for the
   'pitch' in a sequence (SEQU).


************************************
*** Signal Design Function Types ***
************************************


   In order to design your own signal type, you will have to write six
   functions to be linked into your program. They are described below.

   See fnptr.txt for help with function pointer types such as these.


typedef void *(*DUH_LOAD_SIGNAL)(DUH *duh, DUMBFILE *file);

   Write a function conforming to this type which loads or creates all the
   data required by your signal. You may use this, for instance, to load a
   sample. You will be reading directly from the DUH file, so make sure you
   read exactly the quantity of data stored in the file for this signal. You
   should allocate memory (multiple blocks if you like), and return a pointer
   referencing all these data. Your data will be stored, and provided
   whenever this signal is handled. Be careful about using global data, since
   your load_signal function will be called more than once for different
   signals of the same type (e.g. for signals which are both samples but are
   different samples).

   On failure you should return NULL. Please take the possibility of failed
   memory allocation seriously, especially when loading large quantities of
   data. You should make sure all allocated memory is freed before you return
   with failure. See the standard signal types for examples of how to do this
   elegantly using your unload_signal function. On failure, you need not
   worry about reading the right quantity of data from the file.

   Do not close the file under any circumstances.

   Often you will write this function before you have written any code to
   write a DUH file using this signal. If this is the case, don't worry; you
   can write this function free-style, and then the function itself will
   serve as a file format reference when you come to create the file.

   Note that a null return value will always be interpreted as failure. If
   this type of signal does not need to load any data, you should not provide
   a load_signal function at all. The absence of this function will convey
   the intended message to the library.


typedef void *(*DUH_START_SAMPLES)(
   DUH *duh,
   void *signal,
   int n_channels,
   long pos
);

   Write a function conforming to this type. Every time your signal is
   required to produce some output, this will be called first. The 'signal'
   parameter is the same one you returned from your load_signal function. The
   'n_channels' parameter specifies how many channels you will need to render
   sound in. In general you should support one or two, although sometimes
   only one is necessary, depending on how your signal is used. If you can
   write generic code to support more channels, then do so. Store the number
   of channels for use later, unless you're only permitting one value.

   If your signal does not support more than two channels, you are
   responsible for making sure it is never invoked with more than two;
   likewise if your signal only supports one, then make sure it is never
   invoked with more than one. In general you only need to make sure the DUH
   file is never rendered with too many channels (see duh_start_renderer()
   and al_start_duh()). It may be prudent to use Allegro's ASSERT() macro to
   catch bugs of this kind.

   The 'pos' parameter specifies where to start, 65536 being one second into
   the signal. Even if you do not intend to start in the middle of the signal
   yourself, you should support this as it will be used when a DUH is not
   played from the beginning.

   This function should make the necessary preparations in order to render a
   string of samples. All your preparations should be stored in allocated
   memory, to which you return a pointer; this allows several instances of
   the same signal to play at the same time. You will not be given the 'duh'
   and 'signal' parameters again, so you must store them if you need them in
   the render_samples or free_samples functions.

   Once again, on the rare occasions on which you do not need such data (e.g.
   a white noise signal), you should not provide a start_samples function. A
   null return code is interpreted as failure, and your music will be lacking
   notes (which is better than crashing on undetected memory allocation
   failure).


typedef void (*DUH_SET_PARAMETER)(
   void *sampinfo,
   unsigned char id, long value
);

   Some signals can take parameters. For example, a stereo pan signal allows
   you to specify the position at which to source the sample, and you might
   want to choose the cut-off frequency for a low-pass filter. Such
   parameters are set on an instance-by-instance basis, not globally.
   Therefore, if you have any parameters, you should place default values for
   them in the data you initialise in start_samples. Then you should write a
   function conforming to this type, and have it change the parameters when
   the correct IDs are passed (see below).

   Each of your parameters should be identified by a single byte, which will
   be passed in the 'id' parameter. When one of your parameters is correctly
   identified by id, you should change the parameter's value in the data you
   returned from your start_samples function. These data are available via
   the 'sampinfo' parameter. If you do not recognise the ID passed, you may
   wish to log the situation using Allegro's TRACE() macro (which will
   compile away to nothing unless you define DEBUGMODE), since it signifies a
   fault in the DUH file.

   Take some care in deciding what should be a parameter and what should be
   arranged through the use of separate signals (of the same type). For
   example, if you are doing a low-pass filter, you will need a source signal
   on which to apply the filter; this signal's index should be loaded by the
   load_signal function, so you will create separate filter signals for the
   separate source signals. However, the cut-off frequency for the filter is
   more suitably stored in a parameter. Here are some guidelines:

      Continuous values, or discrete quantities, can be stored in parameters.
      Examples: filter cut-off, echo time, stereo pan position.

      Discrete indices or references should be loaded by load_signal.
      Examples: references to other signals.

   Another way of looking at it is as follows:

      If a value could be changed halfway through playing the signal, then
      consider storing it in a parameter.

      If a value needs to be known by the start_samples function and cannot
      change later, then consider loading it in the load_signal function.

   If your signal has no parameters, do not provide a set_parameters
   function. Attempts to change parameters for this signal will not cause a
   crash, but will cause the operation to be logged using Allegro's TRACE()
   macro if the debugging library is used to play the DUH.


typedef long (*DUH_RENDER_SAMPLES)(
   void *sampinfo,
   float volume, float delta,
   long size, sample_t **samples
);

   Write a function conforming to this type. It should render a series of
   samples into the array of buffers provided (see below). You are passed
   'sampinfo' as returned by start_samples. The 'samples' parameter points to
   an array of buffers, one for each channel. You can get a pointer to the
   buffer for channel #n with:

      sample_t *buffer = samples[n];

   As you can see, samples are of type sample_t, which is typedeffed as a
   signed int (32 bits). Your waveform should be centred on 0 and should peak
   at no more than +/- 32768, or +/- 0x8000, given a volume of 1.0f. If your
   waveform goes higher, it will be clipped later on (you do not need to test
   for this yourself). Please do not read any special meaning into the volume
   parameter; it is simply a factor to be multiplied into each sample. If you
   wish to adjust the tone, use a parameter (see DUH_SET_PARAMETER).

   In this function, you should render 'size' samples into the buffer for
   each channel. Return the number of samples actually rendered, which will
   be fewer than 'size' if you run out of samples (e.g. if you reach the end
   of a non-looping sample). Never stop short unless the signal ends, and do
   not overrun under any circumstances. Update the 'sampinfo' data so that
   the samples generated by the next call to render_samples will continue
   seamlessly from the samples generated this time.

   The delta parameter governs the speed at which your signal will play back.
   Higher values of delta should cause the speed and pitch to increase, as
   with duh_render(). In general, if delta is 1.0f then your waveform should
   be suitable for playback at a sampling rate of 65536 Hz. If you wish to
   bend this rule, then be careful - usually it is wiser to adjust your
   thinking in other parts.

   For example, consider a sample recorded at 44100 Hz, stored in a sample
   signal. If a delta of 1.0f is passed to this signal's render function, the
   output will be exactly the same as the original sample - a signal at
   44100 Hz. So it breaks the rule.

   Ah, but it doesn't. Instead, it is wiser to consider the sample stored in
   the DUH file as being sampled at 65536 Hz. Once we consider this, it is
   clear that the output is suitable for playback at 65536 Hz, so we do not
   break the rule. Now it is a simple matter of adjusting all the pitches in
   the sequence to compensate for this. See duhtech.txt for details on how to
   do this when you are writing the DUH file.

   Note that the position that was passed to the start_samples function is
   65536 for one second into the signal as played with delta 1.0f. So, with a
   sample signal, this position is interpreted as the index of the sample on
   which to start. If delta is 2.0, only half a second of the resultant
   output will have been skipped.

   Once again, please do not read any special meaning into the delta
   parameter. The effect of doubling delta should be virtually the same as
   the effect of not doubling delta but resampling the output. If you wish to
   adjust the tone for different pitches, use a parameter.

   This function is compulsory. Silent signals only increase processor and
   memory usage, and we are not Microsoft-Intel evangelists.


typedef void (*DUH_END_SAMPLES)(void *sampinfo);

   Write a function conforming to this type. It will be called when a signal
   stops playing. The parameter points to the data you returned from the
   start_samples function, and here you should simply free the memory up.

   This function should be provided, always if, and only if, start_samples is
   present. The debugging library will check you get this right.


typedef void (*DUH_UNLOAD_SIGNAL)(void *signal);

   Write a function conforming to this type. It will be called when the DUH
   is removed from memory. It should free all memory allocated by your
   load_signal function. The parameter is the same pointer that you returned
   from load_signal.

   If you only ever use this signal type with make_duh(), and not with
   dumb_register_sigtype(), then the following does not apply.

   If load_signal is present, unload_signal should also be present; if you
   did not provide a load_signal function, then you should not provide an
   unload_signal function either. The debugging library will check you get
   this right.


**********************************
*** Signal Design Registration ***
**********************************


void dumb_register_sigtype(DUH_SIGTYPE_DESC *desc);

   When you have written all the functions that represent your signal, you
   will need to register them with the library before it will be able to load
   your DUH file. Use this function. The DUH_SIGTYPE_DESC struct contains the
   following fields:

      long type;
      DUH_LOAD_SIGNAL    load_signal;
      DUH_START_SAMPLES  start_samples;
      DUH_SET_PARAMETER  set_parameter;
      DUH_RENDER_SAMPLES render_samples;
      DUH_END_SAMPLES    end_samples;
      DUH_UNLOAD_SIGNAL  unload_signal;

   You need to create a DUH_SIGTYPE_DESC struct and pass its pointer to
   dumb_register_sigtype(). The struct must be in permanent memory. In other
   words, it must be either global or static, and you should not modify it
   later. DUMB will not make its own copy.

   'type' should be a four-character string encoded with DUMB_ID(), for
   example DUMB_ID('M','E','O','W'). By convention it should be upper case,
   and padded with spaces if you do not use all four characters. However, you
   do not have to stick to this.

   If you are not providing a function, specify NULL for the corresponding
   function pointer.


**********************************
*** Signal Rendering Functions ***
**********************************


   When you are designing your own signals, you will often want to retrieve
   samples from another signal in the DUH. This signal's index ('sig') should
   be loaded by your load_samples function. You can use the following
   functions to obtain the samples.


DUH_SIGNAL_SAMPINFO *duh_signal_start_samples(
   DUH *duh, int sig, int n_channels, long pos
);

   Specifies where you want to start rendering. This function returns a
   DUH_SIGNAL_SAMPINFO struct, which you need to pass to the other functions.
   You can use as many DUH_SIGNAL_SAMPINFOs at once as you like.

   Pass the DUH and the index of the signal whose samples you want to obtain
   ('sig'). Specify how many channels you want, and where you want to start
   rendering (65536 represents one second).

   There is no special need to check that this function succeeds. The other
   functions are safe to call with null pointers. However, checking the
   return value can make your code more efficient.

   Be sure to call duh_signal_end_samples() when you've finished.


void duh_signal_set_parameter(
   DUH_SIGNAL_SAMPINFO *signal_sampinfo,
   unsigned char id, long value
);

   Sets a parameter for the signal whose samples are being rendered by
   signal_sampinfo.




   Calls the set_parameter function for the instance started by
   duh_signal_start_samples of the signal whose details you passed to that
   function. Exactly what this does depends on the signal in question.





   THIS IS NOT VERY HELPFUL. REFER TO A FILE?








long duh_signal_render_samples(
   DUH_SIGNAL_SAMPINFO *signal_sampinfo,
   float volume, float delta,
   long size, sample_t **samples
);

   Renders 'size' samples of the signal for which signal_sampinfo was set up.
   See duh_render() and DUH_RENDER_SAMPLES for details on the 'volume' and
   'delta' parameters. This function will return the number of samples
   generated, which will be fewer than 'size' if the signal ends.

   Sometimes you can pass the array of sample buffers which was passed to
   your function, and process the data in place. Other times you will have to
   set up the array of sample buffer pointers yourself, making sure each
   buffer can hold 'size' samples. Below is some code to do that. Note that
   we prefix some variable names with sub-, so they don't clash with the
   parameters to the function that would typically contain this code.

      sample_t **subsamples;
      int n;
      long subsize;

      subsamples = malloc(n_channels * sizeof(*subsamples));

      if (!subsamples)
         return 0;

      subsamples[0] = malloc(size * n_channels * sizeof(*subsamples[0]));

      if (!subsamples[0]) {
         free(subsamples);
         return 0;
      }

      for (n = 1; n < n_channels; n++)
         subsamples[n] = subsamples[n-1] + size;

      subsize = signal_render_samples(
         subsampinfo,
         volume, delta,
         size, subsamples
      );

      /* Process the samples here. */

      free(subsamples[0]);
      free(subsamples);

      return subsize;


void duh_signal_end_samples(DUH_SIGNAL_SAMPINFO *signal_sampinfo);

   Call this when you have finished with a DUH_SIGNAL_SAMPINFO struct. It
   will free all memory used by the struct.


**************************
*** Resampling Helpers ***
**************************


   The DUH player library provides a versatile resampling system. The sample
   signal type uses it, and it is available for use in any of your signals.

   Be warned that the resampler may overrun the memory you specify by up to
   DUMB_EXTRA_SAMPLES samples, so you must allocate these samples and set
   them to appropriate values when you load your signal. Generally, if you
   are going to loop, set them to reflect the samples at the loop start
   point; if you are not going to loop, set them to zero.

   DUMB_EXTRA_SAMPLES is defined as follows:

      #define DUMB_EXTRA_SAMPLES 3


extern int resampling_quality;

   Allows you to control the quality of all resampling that takes place. This
   may be set to any value from 0 to 4. Higher values will sound better, but
   lower values will use up less processor time.

                   |     --___
      0 - Aliasing |__---     __
                   |            ___--

                            |  __
      1 - Linear resampling | /  \  /\
                            |/    \/  \__

      2 - Linear resampling / linear average

      3 - Quadratic / linear average

      4 - Cubic / linear average

   Level 0 has very noticeable unwanted overtones. It will occasionally
   produce satisfactory results for noisy signals, but usually you will
   want to pay for the extra processor time (which isn't much) and go
   for Level 1.

   Levels 1 and 2 are already pretty good. When resampling down a few
   octaves, however, you will begin to notice unwanted high frequencies.
   These can be eliminated by switching to Levels 3 or 4.

   When Level 2 or higher are selected, a linear average function is used for
   speeding the wave up. Instead of skipping samples, all samples in the
   interval will be averaged. The interval is also smoothed at the edges.
   This will be especially beneficial when increasing the pitch by several
   octaves.

   Levels 3 and 4 are both smooth curves to the eye. They both give
   extremely good performance, but you may sometimes notice the difference
   when reducing the pitch of a sample by several octaves, where Level 3
   may exhibit unwanted high frequencies.








   NOTE: WHAT IS THE DEFAULT? (2 at the moment, but might change. Config?)













long dumb_resample(
   sample_t *src, long *_src_pos, int *_src_subpos,
   long src_start, long src_end,
   sample_t *dst, long dst_size,
   float delta, int *_dir,
   DUMB_RESAMPLE_PICKUP pickup, void *pickup_data
);

   This is the resampling function. It takes an array of source samples and
   fills as much of the destination array as it can for you. Its operation is
   quite complicated, so pay attention.

   All the parameters prefixed with an underline (_) are pointers to the
   fields they describe. This means the function can modify the variables you
   pass, but you have to prefix the variable with an ampersand (&) when
   passing it. If you get warnings about ints being converted to pointers
   without casts, you have most likely forgotten an ampersand somewhere.

   'src' points to the source sample data. It should point to the beginning
   of the sample, even if you are starting your resampling from somewhere in
   the middle. Do not break this rule unless you know what you're doing.

   '_src_pos' and '_src_subpos' together represent the current position in
   the sample. When you first call the function, they represent the starting
   position. Once the function has done its work, they will represent the
   point at which the resampling stopped. '_src_pos' is measured in samples.
   '_src_subpos' represents how far between samples we are, and ranges from
   0 to 65535 - so if _src_subpos is 65535, we are very nearly on to the next
   sample.

   Once _src_pos and _src_subpos have been modified by this function, you can
   pass them to the function again and the resampling will continue
   seamlessly. This is important, as you will hardly ever get to render all
   your samples in one go. Typically you will make one call to this function
   from within your render_samples function.

   This function does not need to know the size of the source buffer as such.
   Instead, it knows src_start and src_end, which are start and end points.
   The end point actually points to the first sample not to use, following
   the usual start-inclusive end-exclusive convention. In general, the
   resampling will stop when src_pos tries to pass one of these.

   WARNING: dumb_resample() may read up to DUMB_EXTRA_SAMPLES samples beyond
            src_end. Make sure the memory belongs to you.

   The _dir parameter should be either 1 or -1, and tells this function
   whether to go forwards or backwards respectively through the source sample
   buffer. If _dir is 0, nothing will happen - resampling has stopped
   permanently. Any other values of _dir will have unpredictable results, and
   the debugging library will abort if you try to use them.

      If _dir is 1, then _src_pos will only be tested against src_end.
      If _dir is -1, then _src_pos will only be tested against src_start.

   That means you can set src_start and src_end to your loop points, and
   start playing from the beginning of the sample. The resampling will
   proceed unimpeded while _src_pos is outside the loop section, provided it
   is advancing towards the loop section. The sample signal makes extensive
   use of this capability.

   The output waveform will be rendered into the buffer pointed to by dst.
   Up to dst_size samples will be generated. Fewer will be generated if the
   resampling stops permanently. The number generated will be returned. You
   can find out if resampling ended by testing the value of your direction
   variable (pointed to by _dir); if it is 0, then resampling stopped
   permanently.

   If you pass NULL for dst, no samples will be generated. However, _src_pos,
   _src_subpos and _dir will be updated as normal. The pick-up function will
   be called as necessary; you should pass the usual 'src' parameter so it
   can be passed to your pick-up function. (See below for information on
   pick-up functions.) The function returns the number of samples that would
   have been generated, had dst pointed somewhere. The operation of
   do_resample() when dst is NULL is exactly the same as that when dst points
   somewhere. However, do_resample() is much faster in this case; you should
   use it when volume is 0, or when you are culling quieter samples to gain
   execution speed.

   If delta is 1.0f, the destination is the same speed as the source. Greater
   values cause the sample to speed up, and lesser values cause the sample to
   slow down. delta should always be positive.

   We have covered most of the parameters. There are only two left - pickup
   and pickup_data. The simplest usage is to set these both to NULL. Then, if
   resampling stops, it stops permanently. Use this if your sample will not
   loop.

   Alternatively, you may wish to write a pick-up function. Your pick-up
   function will take control whenever _src_pos tries to pass the start or
   end points. You can use it for looping, or for a multitude of other tasks.
   The typedef is as follows:

      typedef int (*DUMB_RESAMPLE_PICKUP)(
         sample_t *src, long *_src_pos, int *_src_subpos,
         long *_src_start, long *_src_end,
         int dir,
         void *data
      );

   You are passed the source buffer, in case you want to fill it with new
   samples. You have the _src_pos, _src_subpos, _src_start and _src_end
   pointers, should you need to change the values to which they point. You
   also have dir, but note that it is not a pointer. Instead, you should
   return the new direction, or 0 to stop the resampling permanently.

   The data parameter is a copy of pickup_data as passed to dumb_resample().
   This is typically used to give you access to your 'sampinfo' parameter,
   passed to your render_samples function.

   BEWARE: you must refer to and change src_pos, src_subpos, src_start and
           src_end through the parameters passed. Do not mistakenly use their
           equivalents in your struct instead. The equivalents in your struct
           will eventually be updated, but they will not be accurate during
           the pick-up function.

   On entry to this function, src_pos and src_subpos will have overrun by a
   small amount. When changing them, you should preserve this overrun. The
   following examples will do this for you:

      If you are executing a simple loop, subtract (or add) the difference
      between the loop points from (or to) src_pos. Do not simply set src_pos
      to the other loop point.

         *_src_pos -= *_src_end - *_src_start;

      If you are executing a ping-pong loop, you need to reflect the pointer
      off the loop boundary. To reflect off the loop end point:

         *_src_pos = (*_src_end << 1) - 1 - *_src_pos;
         *_src_subpos ^= 65535;
         dir = -1;

      To reflect off the loop start point:

         *_src_pos = (*_src_start << 1) - 1 - *_src_pos;
         *_src_subpos ^= 65535;
         dir = 1;

      In each case, don't forget to return the new direction correctly!

   An ideal example of a pick-up function can be found in src/sample.c.

   Of course there is more that can be done with a pick-up function. Enjoy
   experimenting!


************************
*** DUH Construction ***
************************


DUH *make_duh(
   long length,
   int n_signals,
   DUH_SIGTYPE_DESC *desc[],
   void *signal[]
);

   Constructs a DUH from its component parts. Use this function if you are
   writing a function to load a music file format other than .duh. Indeed,
   this function is used internally to load IT files.

   Before you call this function, you must do some preparation. Your DUH will
   contain a fixed number of signals; pass this number as n_signals. Each
   signal will have a DUH_SIGTYPE_DESC struct, and you pass an array of
   pointers to these. The DUH_SIGTYPE_DESC struct contains the following
   fields:

      long type;
      DUH_LOAD_SIGNAL    load_signal;
      DUH_START_SAMPLES  start_samples;
      DUH_SET_PARAMETER  set_parameter;
      DUH_RENDER_SAMPLES render_samples;
      DUH_END_SAMPLES    end_samples;
      DUH_UNLOAD_SIGNAL  unload_signal;

   The structs must be in permanent memory, i.e. either global or static, and
   not modified later; however, the array of pointers can be destroyed as
   soon as the function returns.

   The values of 'type' and 'load_signal' are irrelevant; set them to 0 and
   NULL respectively, unless you are using the DUH_SIGTYPE_DESC struct
   elsewhere.

   Because 'load_signal' is never used, you must provide an array of pointers
   to the data that 'load_signal' would otherwise have returned. This will
   be passed to the other functions as 'signal'. Once again, the array of
   pointers can be destroyed as soon as the function returns. As for the
   pointers themselves:

   If an 'unload_signal' function is provided, then that will be used to
   deallocate the pointer when you finally destroy the DUH. If the function
   is not provided, then you are responsible for deallocating any memory
   referenced by that pointer yourself. This applies individually to each
   signal.

   If this function fails and returns NULL, then any 'unload_signal'
   functions you provided will have been called. You do not have any extra
   work to do if it fails.


********************************
*** Allegro Packfile Support ***
********************************


void dumb_register_packfiles(void);

   This function registers the Allegro PACKFILE input module for use by
   DUMBFILEs. PACKFILE structs and their corresponding functions, as defined
   by Allegro's header file allegro.h, will be used internally for all
   DUMBFILE input (unless opened with dumbfile_open_ex()).

   This must be called before dumbfile_open() is used, or else an alternative
   system must be registered (see register_dumbfile_system() and
   dumb_register_stdfiles()).


DUMBFILE *dumbfile_open_packfile(PACKFILE *p);

   If you have an Allegro PACKFILE struct representing an open file, you can
   call this if you wish to read from it using a DUMBFILE. This is useful
   when you need to pass a DUMBFILE struct to a library function, to read an
   embedded music file for example. When you close the DUMBFILE, you can
   continue using the PACKFILE struct to read what follows the embedded data.


***********************************************
*** Allegro Datafile Registration Functions ***
***********************************************


void register_dat_duh(void);

   If you wish to put a DUH file in an Allegro datafile, you must use "DUH "
   for the type. The grabber will have a box for the type when you insert a
   new object. The grabber will treat the DUH file as binary data, which
   means the datafile will contain an exact copy of the DUH file on disk.





   TODO: make it possible to choose the type...





   You must then call register_dat_duh() in your program before you load the
   datafile. Once you've done this, you'll be able to access the DUH using
   the usual datafile[n].dat notation. You do not need to call unload_duh()
   on this DUH; unload_datafile() will do that for you.

   If you need to check the type of the object for whatever reason, you can
   use DAT_DUH, defined as follows:

      #define DAT_DUH DAT_ID('D','U','H',' ')

   The following example iterates through all the DUHs in disan.dat:

      DATAFILE *dat;
      int n;

      register_dat_duh();
      dat = load_datafile("disan.dat");

      for (n = 0; dat[n].type != DAT_END; n++) {
         if (dat[n].type == DAT_DUH) {
            DUH *duh = dat[n].dat;
            /* Insert code here to play 'duh' or whatever you want to do. */
         }
      }

      unload_datafile(dat);


void register_dat_it(void);

   Inserting an IT file in an Allegro datafile is the same as inserting a DUH
   file, except the type has to be "IT  ", the registration function is
   register_dat_it(), and the following definition is available for use
   instead of DAT_DUH:

      #define DAT_IT DAT_ID('I','T',' ',' ')

   Once the datafile is loaded, the 'dat' field indeed points to a DUH
   struct. There are no differences other than those listed above.


*************************************
*** Allegro DUH Playing Functions ***
*************************************


   The functions in this section allow you to play back a DUH through
   Allegro's sound system. You must call Allegro's install_sound() function
   before you use them.


AL_DUH_PLAYER *al_start_duh(
   DUH *duh, int n_channels, long pos, float volume, long bufsize, int freq
);

   Starts playing the specified DUH.

   An AL_DUH_PLAYER represents one instance of the DUH playing. If you wish,
   you can have two or more AL_DUH_PLAYERs going at the same time, for the
   same DUH or for different ones. Each uses one of Allegro's audio streams
   and hence one voice.

   At present, n_channels can either be 1 or 2 for monaural or stereo
   respectively. If you use the debugging library, your program will abort if
   other values are passed; otherwise weird things will happen.

   The DUH will start playing from position 'pos'. 0 represents the start of
   the DUH, and 65536 represents one second. Unlike other music systems, DUMB
   will always make sure every note is there right from the start, provided
   any custom signal types are designed properly. In other words, you can
   start a DUH at a point halfway through a long note, and you will still
   hear the long note.

   The volume is a float. 1.0f is the pseudo-maximum. If you pass 1.0f, any
   properly designed DUH file will play nice and loud, but will not clip. You
   can pass a greater volume if you like, but be prepared for clipping to
   occur. Of course you can pass smaller values to play the DUH more quietly,
   and this will also resolve clipping issues in badly designed DUH files.

   You will need to pass the AL_DUH_PLAYER to other functions when you need
   to stop or pause the DUH, change its volume, or otherwise modify the way
   it is playing. You will also need to pass it to al_poll_duh() at regular
   intervals; if the sound is choppy, try calling al_poll_duh() more often.

   'bufsize' is the number of samples that will be rendered at once. 1024 is
   a suitable value for most purposes.





   NOTE: IS THAT TRUE ON ALL SYSTEMS?





   The greater this is, the less often you will have to call al_poll_duh() -
   but when al_poll_duh() decides to fill the buffer, it will take longer
   doing so. If your game exhibits regular brief freezes, try reducing the
   buffer size. If the sound is choppy, however, you may have to increase it.

   'freq' specifies the sampling frequency at which the DUH should be
   rendered. At present there is no (official and portable) way of knowing
   the frequency at which Allegro is mixing - but if you do know that
   frequency, passing it here will give the highest quality sound. If you
   reduce it, the DUH will sound less crisp but use less processor time.

   When you have finished, you must pass the AL_DUH_PLAYER to al_stop_duh()
   to free up memory. Do not destroy the DUH beforehand.

   There is no real need to check the return value from this function. The
   other functions can be called safely with null pointers, so if there is a
   problem, your music will simply not play.


void al_stop_duh(AL_DUH_PLAYER *dp);

   This will stop an AL_DUH_PLAYER. You must call this when you have finished
   with it, before destroying the DUH. The pointer will no longer be valid on
   return from this function.


void al_pause_duh(AL_DUH_PLAYER *dp);

   This will pause an AL_DUH_PLAYER. Use al_resume_duh() when you want it to
   continue.


void al_resume_duh(AL_DUH_PLAYER *dp);

   Causes a paused AL_DUH_PLAYER to resume playing (see al_pause_duh()).


void al_duh_set_volume(AL_DUH_PLAYER *dp, float volume);

   This will set the volume of an AL_DUH_PLAYER. See al_start_duh() for
   details on the volume parameter.


int al_poll_duh(AL_DUH_PLAYER *dp);

   An AL_DUH_PLAYER is not interrupt-driven. That means it will not play by
   itself. You must keep it alive from your main program. Call this function
   at regular intervals. If the sound crackles, try calling it more often.
   (There is nothing you can do if Windows decides to play with the hard
   disk; that will make your sound crackle no matter what you do.)

   Normally this function will return zero. However, if it returns nonzero,
   that means the AL_DUH_PLAYER will not generate any more sound. Indeed the
   underlying audio stream and DUH_RENDERER have been destroyed. When this
   happens, you can call al_stop_duh() whenever you wish - but you do not
   have to. Note that this function will wait two buffers' worth of samples
   before taking this action, allowing Allegro to mix the trailing sound
   before the audio stream is destroyed. In other words, your music will not
   be cut off prematurely.

   In case you were wondering, it is definitely not safe to call
   al_poll_duh() from an interrupt context. Not only is no part of DUMB
   locked in memory, but many signals and many core library functions
   allocate and free their memory on a call-by-call basis! Remember that any
   disk access that occurs in interrupt context is likely to crash the
   machine. (This limitation only applies to DOS at present, and is due to
   the fact that the DOS file access functions are not re-entrant.
   Multitasking systems are generally safe. However, even if you do not think
   you are targeting DOS, it is worth considering that calling this function
   regularly from your main loop is likely to be much more efficient than
   relying on task switching to do it for you.)


long al_duh_get_position(AL_DUH_PLAYER *dp);

   Tells you what position an AL_DUH_PLAYER is up to, or -1 if it is invalid
   (perhaps owing to lack of memory). As usual, 65536 is one second. Note
   that this is a whole number, whereas a fractional part is stored
   internally; the sample will not be continuous if you terminate the
   AL_DUH_PLAYER and then reinitiate it with the same position. Furthermore,
   note that Allegro will not have mixed in all the sound up to this point;
   if you wait for this to reach a certain position and then terminate the
   AL_DUH_PLAYER, the sound will cut off too early.








   NOTE: HOW TO GET AROUND THIS?








DUH_RENDERER *al_duh_get_renderer(AL_DUH_PLAYER *dp);

   This function returns to you the DUH_RENDERER underlying the specified
   AL_DUH_PLAYER. It has no practical use in this release, but it
   TODO: Write this. Also see if it should go in howto.txt.


******************
*** Conclusion ***
******************


I conclude that... DUMB is the bestest music player in the world because...
Complete this sentence in fifteen words or fewer... D'OH!


Ben Davis
entheh@users.sf.net
IRC EFnet #dumb
See readme.txt for details on using IRC.