1
1
Fork 0
mirror of https://github.com/QB64-Phoenix-Edition/QB64pe.git synced 2024-09-19 22:35:07 +00:00
QB64-PE/internal/c/parts/audio/extras/hivelytracker/hvl_replay.c

2049 lines
85 KiB
C
Raw Normal View History

2023-01-01 17:26:02 +00:00
/*
** Changes for the 1.4 release are commented. You can do
** a search for "1.4" and merge them into your own replay
** code.
**
** Changes for 1.5 are marked also.
**
** ... as are those for 1.6
**
** ... and for 1.8
*/
#include "hvl_replay.h"
#include <math.h>
2023-01-01 17:26:02 +00:00
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
2023-01-01 17:26:02 +00:00
#define WHITENOISELEN (0x280 * 3)
#define WO_LOWPASSES 0
#define WO_TRIANGLE_04 (WO_LOWPASSES + ((0xfc + 0xfc + 0x80 * 0x1f + 0x80 + 3 * 0x280) * 31))
#define WO_TRIANGLE_08 (WO_TRIANGLE_04 + 0x04)
#define WO_TRIANGLE_10 (WO_TRIANGLE_08 + 0x08)
#define WO_TRIANGLE_20 (WO_TRIANGLE_10 + 0x10)
#define WO_TRIANGLE_40 (WO_TRIANGLE_20 + 0x20)
#define WO_TRIANGLE_80 (WO_TRIANGLE_40 + 0x40)
#define WO_SAWTOOTH_04 (WO_TRIANGLE_80 + 0x80)
#define WO_SAWTOOTH_08 (WO_SAWTOOTH_04 + 0x04)
#define WO_SAWTOOTH_10 (WO_SAWTOOTH_08 + 0x08)
#define WO_SAWTOOTH_20 (WO_SAWTOOTH_10 + 0x10)
#define WO_SAWTOOTH_40 (WO_SAWTOOTH_20 + 0x20)
#define WO_SAWTOOTH_80 (WO_SAWTOOTH_40 + 0x40)
#define WO_SQUARES (WO_SAWTOOTH_80 + 0x80)
#define WO_WHITENOISE (WO_SQUARES + (0x80 * 0x20))
#define WO_HIGHPASSES (WO_WHITENOISE + WHITENOISELEN)
#define WAVES_SIZE (WO_HIGHPASSES + ((0xfc + 0xfc + 0x80 * 0x1f + 0x80 + 3 * 0x280) * 31))
const uint16 lentab[45] = {3, 7, 0xf, 0x1f, 0x3f, 0x7f, 3, 7, 0xf, 0x1f, 0x3f, 0x7f, 0x7f, 0x7f, 0x7f,
0x7f, 0x7f, 0x7f, 0x7f, 0x7f, 0x7f, 0x7f, 0x7f, 0x7f, 0x7f, 0x7f, 0x7f, 0x7f, 0x7f, 0x7f,
0x7f, 0x7f, 0x7f, 0x7f, 0x7f, 0x7f, 0x7f, 0x7f, 0x7f, 0x7f, 0x7f, 0x7f, 0x7f, 0x7f, (0x280 * 3) - 1};
const int16 vib_tab[64] = {0, 24, 49, 74, 97, 120, 141, 161, 180, 197, 212, 224, 235, 244, 250, 253, 255, 253, 250, 244, 235, 224,
212, 197, 180, 161, 141, 120, 97, 74, 49, 24, 0, -24, -49, -74, -97, -120, -141, -161, -180, -197, -212, -224,
-235, -244, -250, -253, -255, -253, -250, -244, -235, -224, -212, -197, -180, -161, -141, -120, -97, -74, -49, -24};
const uint16 period_tab[61] = {0x0000, 0x0D60, 0x0CA0, 0x0BE8, 0x0B40, 0x0A98, 0x0A00, 0x0970, 0x08E8, 0x0868, 0x07F0, 0x0780, 0x0714, 0x06B0, 0x0650, 0x05F4,
0x05A0, 0x054C, 0x0500, 0x04B8, 0x0474, 0x0434, 0x03F8, 0x03C0, 0x038A, 0x0358, 0x0328, 0x02FA, 0x02D0, 0x02A6, 0x0280, 0x025C,
0x023A, 0x021A, 0x01FC, 0x01E0, 0x01C5, 0x01AC, 0x0194, 0x017D, 0x0168, 0x0153, 0x0140, 0x012E, 0x011D, 0x010D, 0x00FE, 0x00F0,
0x00E2, 0x00D6, 0x00CA, 0x00BE, 0x00B4, 0x00AA, 0x00A0, 0x0097, 0x008F, 0x0087, 0x007F, 0x0078, 0x0071};
const int32 stereopan_left[5] = {128, 96, 64, 32, 0};
const int32 stereopan_right[5] = {128, 160, 193, 225, 255};
const int16 filter_thing[2790] = {
-1161, -4413, -7161, -13094, 635, 13255, 2189, 6401, 9041, 16130, 13460, 5360, 6349, 12699, 19049, 25398, 30464, 32512, 32512,
32515, 31625, 29756, 27158, 24060, 20667, 17156, 13970, 11375, 9263, 7543, 6142, 5002, 4074, 3318, 2702, 2178, 1755, 1415,
1141, 909, 716, 563, 444, 331, -665, -2082, -6170, -9235, -13622, 12545, 9617, 3951, 8345, 11246, 18486, 6917, 3848,
8635, 17271, 25907, 32163, 32512, 32455, 30734, 27424, 23137, 18397, 13869, 10429, 7843, 5897, 4435, 3335, 2507, 1885, 1389,
1023, 720, 530, 353, 260, 173, 96, 32, -18, -55, -79, -92, -95, -838, -3229, -7298, -12386, -7107, 13946,
6501, 5970, 9133, 14947, 16881, 6081, 3048, 10921, 21843, 31371, 32512, 32068, 28864, 23686, 17672, 12233, 8469, 5862, 4058,
2809, 1944, 1346, 900, 601, 371, 223, 137, 64, 7, -34, -58, -69, -70, -63, -52, -39, -26, -14,
-5, 4984, -4476, -8102, -14892, 2894, 12723, 4883, 8010, 9750, 17887, 11790, 5099, 2520, 13207, 26415, 32512, 32457, 28690,
22093, 14665, 9312, 5913, 3754, 2384, 1513, 911, 548, 330, 143, 3, -86, -130, -139, -125, -97, -65, -35,
-11, 6, 15, 19, 19, 16, 12, 8, 6877, -5755, -9129, -15709, 9705, 10893, 4157, 9882, 10897, 19236, 8153,
4285, 2149, 15493, 30618, 32512, 30220, 22942, 14203, 8241, 4781, 2774, 1609, 933, 501, 220, 81, 35, 2, -18,
-26, -25, -20, -13, -7, -1, 2, 4, 4, 3, 2, 1, 0, 0, -1, 2431, -6956, -10698, -14594,
12720, 8980, 3714, 10892, 12622, 19554, 6915, 3745, 1872, 17779, 32512, 32622, 26286, 16302, 8605, 4542, 2397, 1265, 599,
283, 45, -92, -141, -131, -93, -49, -14, 8, 18, 18, 14, 8, 3, 0, -2, -3, -2, -2,
-1, 0, 0, -3654, -8008, -12743, -11088, 13625, 7342, 3330, 11330, 14859, 18769, 6484, 3319, 1660, 20065, 32512, 30699,
21108, 10616, 5075, 2425, 1159, 477, 196, 1, -93, -109, -82, -44, -12, 7, 14, 13, 9, 4, 0,
-2, -2, -1, -1, 0, 0, 0, 0, 0, 0, -7765, -8867, -14957, -5862, 13550, 6139, 2988, 11284, 17054,
16602, 6017, 2979, 1489, 22351, 32512, 28083, 15576, 6708, 2888, 1243, 535, 188, 32, -47, -64, -47, -22, -3,
7, 8, 5, 3, 0, -1, -1, -1, 0, 0, 0, 0, 0, 0, 0, 0, 0, -9079, -9532,
-16960, -335, 13001, 5333, 2704, 11192, 18742, 13697, 5457, 2703, 1351, 24637, 32512, 24556, 10851, 4185, 1614, 622, 184,
15, -57, -59, -34, -9, 5, 8, 6, 2, 0, -1, -1, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, -8576, -10043, -18551, 4372, 12190, 4809, 2472, 11230, 19803, 11170, 4953, 2473, 1236, 26923,
32512, 20567, 7430, 2550, 875, 212, 51, -30, -43, -25, -6, 3, 5, 3, 1, 0, -1, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, -6960, -10485, -19740, 7864, 11223, 4449, 2279,
11623, 20380, 9488, 4553, 2280, 1140, 29209, 31829, 16235, 4924, 1493, 452, 86, -7, -32, -20, -5, 2, 3,
2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
-4739, -10974, -19831, 10240, 10190, 4169, 2114, 12524, 20649, 8531, 4226, 2114, 1057, 31495, 29672, 11916, 3168, 841, 121,
17, -22, -18, -5, 2, 2, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, -2333, -11641, -19288, 11765, 9175, 3923, 1971, 13889, 20646, 8007, 3942, 1971,
985, 32512, 27426, 8446, 1949, 449, 45, -11, -16, -5, 1, 1, 1, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 29, -12616, -17971, 12690, 8247,
3693, 1846, 15662, 20271, 7658, 3692, 1846, 923, 32512, 25132, 6284, 1245, 246, -71, -78, -17, 8, 7, 1,
-1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 2232, -14001, -15234, 13198, 7447, 3478, 1736, 17409, 19411, 7332, 3472, 1736, 868, 32512, 22545, 4352, 731,
18, -117, -40, 8, 9, 2, -1, -1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 4197, -15836, -11480, 13408, 6791, 3281, 1639, 19224, 18074, 6978,
3276, 1639, 819, 32512, 19657, 2706, 380, -148, -86, 2, 13, 3, -2, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 5863, -17878, -9460,
13389, 6270, 3104, 1551, 20996, 16431, 6616, 3102, 1551, 776, 32512, 16633, 1921, 221, -95, -39, 5, 5, 0,
-1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 7180, -20270, -6194, 13181, 5866, 2946, 1473, 22548, 14746, 6273, 2946, 1473, 737, 32512, 13621,
1263, 116, -53, -15, 4, 2, -1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 8117, -21129, -2795, 12809, 5550, 2804, 1402, 23717,
13326, 5962, 2804, 1402, 701, 32512, 10687, 776, -56, -56, 4, 4, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 8560,
-19953, 508, 12299, 5295, 2675, 1337, 25109, 12263, 5684, 2675, 1338, 669, 32512, 7905, 433, -36, -22, 3, 1,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 8488, -18731, 3672, 11679, 5080, 2558, 1279, 26855, 11480, 5434, 2557, 1279, 639,
32512, 5357, 212, -95, 0, 4, -1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7977, -24055, 6537, 10986, 4883, 2450,
1225, 28611, 10918, 5206, 2450, 1225, 612, 32512, 3131, 83, -35, 2, 1, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 7088, -30584, 9054, 10265, 4696, 2351, 1176, 28707, 10494, 4996, 2351, 1175, 588, 32512, 1920, -155, -13, 4,
-1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 5952, -32627, 11249, 9564, 4519, 2260, 1130, 28678, 10113, 4803, 2260,
1130, 565, 32512, 1059, -73, -1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 4629, -32753, 13199, 8934,
4351, 2175, 1088, 28446, 9775, 4623, 2175, 1087, 544, 32512, 434, -22, 1, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 3132, -32768, 15225, 8430, 4194, 2097, 1049, 30732, 9439, 4456, 2097, 1049, 524, 32512, 75, -6,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1345, -32768, 16765, 8107, 4048, 2025, 1012, 32512, 9112,
4302, 2025, 1012, 506, 32385, 392, 5, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, -706, -32768,
17879, 8005, 3913, 1956, 978, 32512, 8843, 4157, 1957, 978, 489, 31184, 1671, 122, 10, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, -3050, -32768, 18923, 8163, 3799, 1893, 946, 32512, 8613, 4022, 1893, 945, 473, 29903,
3074, 316, 52, 11, 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, -5812, -32768, 19851, 8626, 3739, 1833, 917,
32512, 7982, 3889, 1833, 916, 459, 28541, 4567, 731, 206, 66, 23, 8, 1, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
-9235, -32768, 20587, 9408, 3841, 1784, 889, 32512, 6486, 3688, 1776, 889, 447, 27099, 6112, 1379, 313, 135, 65,
33, 17, 7, 4, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
2, 2, 2, 2, 2, 2, 2, -12713, 1188, 1318, -1178, -4304, -26320, -14931, -1716, -1486, 2494, 3611, 22275,
27450, -31839, -29668, -26258, -21608, -15880, -9560, -3211, 3138, 9369, 15281, 20717, 25571, 29774, 32512, 32512, 32512, 32512, 32512,
32512, 32512, 32512, 32512, 32512, 32512, 32512, 32512, 32748, 32600, 32750, 32566, 32659, 32730, 8886, 1762, 506, -1665, -12112,
-24641, -8513, -2224, 247, 3288, 9926, 25787, 28909, -31048, -27034, -20726, -12532, -3896, 4733, 13043, 20568, 27010, 32215, 32512,
32512, 32512, 32512, 32512, 32512, 32512, 32762, 32696, 32647, 32512, 32665, 32512, 32587, 32638, 32669, 32681, 32679, 32667, 32648,
32624, 32598, 6183, 2141, -630, -2674, -21856, -18306, -5711, -2161, 2207, 4247, 17616, 26475, 29719, -30017, -23596, -13741, -2819,
8029, 18049, 26470, 32512, 32512, 32512, 32512, 32512, 32512, 32512, 32738, 32663, 32612, 32756, 32549, 32602, 32629, 32636, 32628,
32610, 32588, 32564, 32542, 32524, 32510, 32500, 32494, 32492, 3604, 2248, -1495, -5612, -26800, -13545, -4745, -1390, 3443, 6973,
23495, 27724, 30246, -28745, -19355, -6335, 6861, 19001, 28690, 32512, 32512, 32512, 32512, 32512, 32512, 32512, 32512, 32667, 32743,
32757, 32730, 32681, 32624, 32572, 32529, 32500, 32482, 32476, 32477, 32482, 32489, 32497, 32504, 32509, 32513, 7977, 1975, -1861,
-9752, -25893, -10150, -4241, 86, 4190, 10643, 25235, 28481, 30618, -27231, -14398, 1096, 15982, 27872, 32512, 32512, 32512, 32512,
32512, 32734, 32631, 32767, 32531, 32553, 32557, 32551, 32539, 32527, 32516, 32509, 32505, 32504, 32505, 32506, 32508, 32510, 32511,
32512, 32512, 32512, 32511, 14529, 1389, -2028, -14813, -22765, -7845, -3774, 1986, 4706, 14562, 25541, 29019, 30894, -25476, -9294,
8516, 23979, 32512, 32512, 32512, 32512, 32512, 32512, 32708, 32762, 32727, 32654, 32579, 32522, 32490, 32478, 32480, 32488, 32498,
32507, 32512, 32515, 32515, 32514, 32513, 32512, 32510, 32510, 32510, 32510, 17663, 557, -2504, -19988, -19501, -6436, -3340, 4135,
5461, 18788, 26016, 29448, 31107, -23481, -4160, 15347, 30045, 32512, 32512, 32512, 32512, 32512, 32674, 32700, 32654, 32586, 32531,
32498, 32486, 32488, 32496, 32504, 32510, 32513, 32514, 32513, 32512, 32511, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 16286,
-402, -3522, -23951, -16641, -5631, -2983, 6251, 6837, 22781, 26712, 29788, 31277, -21244, 1108, 21806, 32512, 32512, 32512, 32512,
32695, 32576, 32622, 32600, 32557, 32520, 32501, 32496, 32500, 32505, 32509, 32512, 32512, 32512, 32511, 32510, 32510, 32510, 32510,
32510, 32510, 32510, 32510, 32510, 32510, 13436, -1351, -4793, -25948, -14224, -5151, -2702, 7687, 8805, 25705, 27348, 30064, 31415,
-18766, 5872, 26652, 32512, 32512, 32512, 32747, 32581, 32620, 32586, 32540, 32508, 32497, 32499, 32505, 32510, 32512, 32512, 32512,
32511, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 10427, -2162, -7136, -26147, -12195, -4810,
-2474, 8723, 11098, 27251, 27832, 30293, 31530, -16047, 10877, 30990, 32512, 32512, 32512, 32512, 32584, 32571, 32536, 32511, 32502,
32503, 32507, 32510, 32512, 32512, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511,
32511, 7797, -2748, -10188, -25174, -10519, -4515, -2281, 9397, 13473, 27937, 28213, 30487, 31627, -13087, 15816, 32512, 32512, 32512,
32715, 32550, 32560, 32534, 32512, 32505, 32506, 32508, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511,
32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 5840, -3084, -13327, -23617, -9177, -4231, -2116, 9892, 15843, 28292, 28538,
30652, 31710, -9886, 20235, 32512, 32512, 32512, 32512, 32550, 32534, 32514, 32507, 32507, 32510, 32511, 32511, 32511, 32511, 32511,
32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 4592, -3215, -15898, -21856,
-8141, -3958, -1972, 10401, 18229, 28612, 28824, 30796, 31781, -7103, 24037, 32512, 32512, 32745, 32535, 32534, 32517, 32508, 32508,
32509, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510,
32510, 32510, 32510, 3964, -3262, -18721, -20087, -7368, -3705, -1847, 11014, 20634, 28996, 29075, 30920, 31843, -4732, 27243, 32512,
32512, 32648, 32627, 32530, 32495, 32500, 32510, 32512, 32512, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511,
32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 3858, -3404, -21965, -18398, -6801, -3479, -1738, 12009, 22960,
29429, 29294, 31030, 31898, -2281, 30194, 32512, 32512, 32699, 32569, 32496, 32496, 32509, 32513, 32512, 32510, 32510, 32510, 32510,
32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 4177, -3869,
-24180, -16820, -6380, -3280, -1640, 13235, 25035, 29863, 29490, 31128, 31947, 251, 32758, 32512, 32749, 32652, 32508, 32490, 32507,
32513, 32512, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511,
32511, 32511, 32511, 32511, 32511, 4837, -4913, -26436, -15364, -6056, -3103, -1553, 14759, 26704, 30256, 29664, 31215, 31991, 2863,
32512, 32512, 32657, 32580, 32503, 32501, 32510, 32512, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510,
32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 5755, -6290, -27702, -14036, -5788, -2947, -1474,
16549, 27912, 30602, 29821, 31294, 32030, 5555, 32512, 32512, 32592, 32541, 32505, 32507, 32511, 32511, 32510, 32510, 32510, 32510,
32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510,
6898, -8911, -27788, -12841, -5550, -2805, -1403, 18509, 28687, 30906, 29963, 31364, 32066, 8328, 32512, 32512, 32623, 32511, 32502,
32510, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511,
32511, 32511, 32511, 32511, 32511, 32511, 32511, 8107, -11465, -27077, -11789, -5325, -2676, -1339, 19833, 29213, 31179, 30092, 31429,
32098, 11181, 32512, 32512, 32561, 32508, 32508, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511,
32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 9247, -13203, -25808, -10886, -5109,
-2559, -1280, 21060, 29636, 31428, 30209, 31488, 32127, 14114, 32512, 32681, 32529, 32502, 32511, 32511, 32511, 32511, 32511, 32511,
32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511,
32511, 32511, 10252, -16863, -24251, -10137, -4902, -2451, -1226, 21937, 30022, 31656, 30317, 31542, 32154, 17128, 32512, 32581, 32514,
32508, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511,
32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 11032, -22427, -22598, -9535, -4705, -2353, -1177, 20999, 30406, 31867,
30415, 31591, 32179, 20222, 32512, 32591, 32501, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511,
32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 11539, -19778, -20962,
-9060, -4522, -2261, -1131, 19486, 30789, 32061, 30507, 31637, 32201, 23396, 32512, 32535, 32508, 32511, 32511, 32511, 32511, 32511,
32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511,
32511, 32511, 32511, 32511, 11803, -12759, -19353, -8690, -4353, -2177, -1089, 18499, 31165, 32240, 30591, 31678, 32222, 26651, 32512,
32514, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511,
32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 11826, -7586, -17510, -8384, -4196, -2099, -1050, 26861,
31521, 32406, 30669, 31718, 32241, 29986, 32585, 32510, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511,
32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 32511, 11599,
-2848, -15807, -8097, -4051, -2025, -1014, 30693, 31850, 32561, 30743, 31755, 32261, 32512, 32524, 32510, 32510, 32510, 32510, 32510,
32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510,
32510, 32510, 32510, 32510, 32510, 32510, 11037, -5302, -14051, -7770, -3913, -1958, -980, 28033, 32165, 32705, 30810, 31789, 32278,
32512, 32729, 32536, 32513, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510,
32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 10114, -7837, -12293, -7348, -3782, -1894,
-948, 24926, 32473, 32512, 30873, 31819, 32294, 32512, 32512, 32580, 32527, 32515, 32512, 32510, 32510, 32510, 32510, 32510, 32510,
32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510,
32510, 8759, -10456, -10591, -6766, -3638, -1835, -917, 24058, 32600, 32512, 30934, 31850, 32309, 32512, 32512, 32729, 32591, 32537,
32520, 32514, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510,
32510, 32510, 32510, 32510, 32510, 32510, 32510, 32510, 6811, -13156, -9045, -5965, -3421, -1776, -890, 31582, 32246, 32512, 30988,
31878, 32324, 32512, 32512, 32512, 32628, 32573, 32541, 32526, 32518, 32514, 32513, 32512, 32512, 32512, 32512, 32512, 32512, 32512,
32512, 32512, 32512, 32512, 32512, 32512, 32512, 32512, 32512, 32512, 32512, 32512, 32512, 32512, 32512, 4835};
2023-01-01 17:26:02 +00:00
int8 waves[WAVES_SIZE];
uint32 panning_left[256], panning_right[256];
void hvl_GenPanningTables(void) {
uint32 i;
float64 aa, ab;
// Sine based panning table
aa = (3.14159265f * 2.0f) / 4.0f; // Quarter of the way through the sinewave == top peak
ab = 0.0f; // Start of the climb from zero
for (i = 0; i < 256; i++) {
panning_left[i] = (uint32)(sin(aa) * 255.0f);
panning_right[i] = (uint32)(sin(ab) * 255.0f);
aa += (3.14159265 * 2.0f / 4.0f) / 256.0f;
ab += (3.14159265 * 2.0f / 4.0f) / 256.0f;
}
panning_left[255] = 0;
panning_right[0] = 0;
2023-01-01 17:26:02 +00:00
}
void hvl_GenSawtooth(int8 *buf, uint32 len) {
uint32 i;
int32 val, add;
2023-01-01 17:26:02 +00:00
add = 256 / (len - 1);
val = -128;
2023-01-01 17:26:02 +00:00
for (i = 0; i < len; i++, val += add)
*buf++ = (int8)val;
2023-01-01 17:26:02 +00:00
}
void hvl_GenTriangle(int8 *buf, uint32 len) {
uint32 i;
int32 d2, d5, d1, d4;
int32 val;
int8 *buf2;
d2 = len;
d5 = len >> 2;
d1 = 128 / d5;
d4 = -(d2 >> 1);
val = 0;
for (i = 0; i < d5; i++) {
*buf++ = val;
val += d1;
}
*buf++ = 0x7f;
if (d5 != 1) {
val = 128;
for (i = 0; i < d5 - 1; i++) {
val -= d1;
*buf++ = val;
}
}
buf2 = buf + d4;
for (i = 0; i < d5 * 2; i++) {
int8 c;
c = *buf2++;
if (c == 0x7f)
c = 0x80;
else
c = -c;
*buf++ = c;
}
2023-01-01 17:26:02 +00:00
}
void hvl_GenSquare(int8 *buf) {
uint32 i, j;
for (i = 1; i <= 0x20; i++) {
for (j = 0; j < (0x40 - i) * 2; j++)
*buf++ = 0x80;
for (j = 0; j < i * 2; j++)
*buf++ = 0x7f;
}
2023-01-01 17:26:02 +00:00
}
static inline int32 clipshifted8(int32 in) {
int16 top = (int16)(in >> 16);
if (top > 127)
in = 127 << 16;
else if (top < -128)
in = -(128 << 16);
return in;
2023-01-01 17:26:02 +00:00
}
void hvl_GenFilterWaves(const int8 *buf, int8 *lowbuf, int8 *highbuf) {
const int16 *mid_table = &filter_thing[0];
const int16 *low_table = &filter_thing[1395];
int32 freq;
int32 i;
for (i = 0, freq = 25; i < 31; i++, freq += 9) {
uint32 wv;
const int8 *a0 = buf;
for (wv = 0; wv < 6 + 6 + 0x20 + 1; wv++) {
int32 in, fre, high, mid, low;
uint32 j;
mid = *mid_table++ << 8;
low = *low_table++ << 8;
for (j = 0; j <= lentab[wv]; j++) {
in = a0[j] << 16;
high = clipshifted8(in - mid - low);
fre = (high >> 8) * freq;
mid = clipshifted8(mid + fre);
fre = (mid >> 8) * freq;
low = clipshifted8(low + fre);
*highbuf++ = high >> 16;
*lowbuf++ = low >> 16;
}
a0 += lentab[wv] + 1;
}
}
2023-01-01 17:26:02 +00:00
}
void hvl_GenWhiteNoise(int8 *buf, uint32 len) {
uint32 ays;
2023-01-01 17:26:02 +00:00
ays = 0x41595321;
2023-01-01 17:26:02 +00:00
do {
uint16 ax, bx;
int8 s;
2023-01-01 17:26:02 +00:00
s = ays;
2023-01-01 17:26:02 +00:00
if (ays & 0x100) {
s = 0x7f;
2023-01-01 17:26:02 +00:00
if (ays & 0x8000)
s = 0x80;
}
2023-01-01 17:26:02 +00:00
*buf++ = s;
len--;
2023-01-01 17:26:02 +00:00
ays = (ays >> 5) | (ays << 27);
ays = (ays & 0xffffff00) | ((ays & 0xff) ^ 0x9a);
bx = ays;
ays = (ays << 2) | (ays >> 30);
ax = ays;
bx += ax;
ax ^= bx;
ays = (ays & 0xffff0000) | ax;
ays = (ays >> 3) | (ays << 29);
} while (len);
2023-01-01 17:26:02 +00:00
}
void hvl_GenTables(void) {
hvl_GenPanningTables();
hvl_GenSawtooth(&waves[WO_SAWTOOTH_04], 0x04);
hvl_GenSawtooth(&waves[WO_SAWTOOTH_08], 0x08);
hvl_GenSawtooth(&waves[WO_SAWTOOTH_10], 0x10);
hvl_GenSawtooth(&waves[WO_SAWTOOTH_20], 0x20);
hvl_GenSawtooth(&waves[WO_SAWTOOTH_40], 0x40);
hvl_GenSawtooth(&waves[WO_SAWTOOTH_80], 0x80);
hvl_GenTriangle(&waves[WO_TRIANGLE_04], 0x04);
hvl_GenTriangle(&waves[WO_TRIANGLE_08], 0x08);
hvl_GenTriangle(&waves[WO_TRIANGLE_10], 0x10);
hvl_GenTriangle(&waves[WO_TRIANGLE_20], 0x20);
hvl_GenTriangle(&waves[WO_TRIANGLE_40], 0x40);
hvl_GenTriangle(&waves[WO_TRIANGLE_80], 0x80);
hvl_GenSquare(&waves[WO_SQUARES]);
hvl_GenWhiteNoise(&waves[WO_WHITENOISE], WHITENOISELEN);
hvl_GenFilterWaves(&waves[WO_TRIANGLE_04], &waves[WO_LOWPASSES], &waves[WO_HIGHPASSES]);
2023-01-01 17:26:02 +00:00
}
/*
** Waves
*/
void hvl_reset_some_stuff(struct hvl_tune *ht) {
uint32 i;
for (i = 0; i < MAX_CHANNELS; i++) {
ht->ht_Voices[i].vc_Delta = 1;
ht->ht_Voices[i].vc_OverrideTranspose = 1000; // 1.5
ht->ht_Voices[i].vc_SamplePos = ht->ht_Voices[i].vc_Track = ht->ht_Voices[i].vc_Transpose = ht->ht_Voices[i].vc_NextTrack =
ht->ht_Voices[i].vc_NextTranspose = 0;
ht->ht_Voices[i].vc_ADSRVolume = ht->ht_Voices[i].vc_InstrPeriod = ht->ht_Voices[i].vc_TrackPeriod = ht->ht_Voices[i].vc_VibratoPeriod =
ht->ht_Voices[i].vc_NoteMaxVolume = ht->ht_Voices[i].vc_PerfSubVolume = ht->ht_Voices[i].vc_TrackMasterVolume = 0;
ht->ht_Voices[i].vc_NewWaveform = ht->ht_Voices[i].vc_Waveform = ht->ht_Voices[i].vc_PlantSquare = ht->ht_Voices[i].vc_PlantPeriod =
ht->ht_Voices[i].vc_IgnoreSquare = 0;
ht->ht_Voices[i].vc_TrackOn = ht->ht_Voices[i].vc_FixedNote = ht->ht_Voices[i].vc_VolumeSlideUp = ht->ht_Voices[i].vc_VolumeSlideDown =
ht->ht_Voices[i].vc_HardCut = ht->ht_Voices[i].vc_HardCutRelease = ht->ht_Voices[i].vc_HardCutReleaseF = 0;
ht->ht_Voices[i].vc_PeriodSlideSpeed = ht->ht_Voices[i].vc_PeriodSlidePeriod = ht->ht_Voices[i].vc_PeriodSlideLimit =
ht->ht_Voices[i].vc_PeriodSlideOn = ht->ht_Voices[i].vc_PeriodSlideWithLimit = 0;
ht->ht_Voices[i].vc_PeriodPerfSlideSpeed = ht->ht_Voices[i].vc_PeriodPerfSlidePeriod = ht->ht_Voices[i].vc_PeriodPerfSlideOn =
ht->ht_Voices[i].vc_VibratoDelay = ht->ht_Voices[i].vc_VibratoCurrent = ht->ht_Voices[i].vc_VibratoDepth = ht->ht_Voices[i].vc_VibratoSpeed = 0;
ht->ht_Voices[i].vc_SquareOn = ht->ht_Voices[i].vc_SquareInit = ht->ht_Voices[i].vc_SquareLowerLimit = ht->ht_Voices[i].vc_SquareUpperLimit =
ht->ht_Voices[i].vc_SquarePos = ht->ht_Voices[i].vc_SquareSign = ht->ht_Voices[i].vc_SquareSlidingIn = ht->ht_Voices[i].vc_SquareReverse = 0;
ht->ht_Voices[i].vc_FilterOn = ht->ht_Voices[i].vc_FilterInit = ht->ht_Voices[i].vc_FilterLowerLimit = ht->ht_Voices[i].vc_FilterUpperLimit =
ht->ht_Voices[i].vc_FilterPos = ht->ht_Voices[i].vc_FilterSign = ht->ht_Voices[i].vc_FilterSpeed = ht->ht_Voices[i].vc_FilterSlidingIn =
ht->ht_Voices[i].vc_IgnoreFilter = 0;
ht->ht_Voices[i].vc_PerfCurrent = ht->ht_Voices[i].vc_PerfSpeed = ht->ht_Voices[i].vc_WaveLength = ht->ht_Voices[i].vc_NoteDelayOn =
ht->ht_Voices[i].vc_NoteCutOn = 0;
ht->ht_Voices[i].vc_AudioPeriod = ht->ht_Voices[i].vc_AudioVolume = ht->ht_Voices[i].vc_VoiceVolume = ht->ht_Voices[i].vc_VoicePeriod =
ht->ht_Voices[i].vc_VoiceNum = ht->ht_Voices[i].vc_WNRandom = 0;
ht->ht_Voices[i].vc_SquareWait = ht->ht_Voices[i].vc_FilterWait = ht->ht_Voices[i].vc_PerfWait = ht->ht_Voices[i].vc_NoteDelayWait =
ht->ht_Voices[i].vc_NoteCutWait = 0;
ht->ht_Voices[i].vc_PerfList = 0;
ht->ht_Voices[i].vc_RingSamplePos = ht->ht_Voices[i].vc_RingDelta = ht->ht_Voices[i].vc_RingPlantPeriod = ht->ht_Voices[i].vc_RingAudioPeriod =
ht->ht_Voices[i].vc_RingNewWaveform = ht->ht_Voices[i].vc_RingWaveform = ht->ht_Voices[i].vc_RingFixedPeriod = ht->ht_Voices[i].vc_RingBasePeriod =
0;
ht->ht_Voices[i].vc_RingMixSource = NULL;
ht->ht_Voices[i].vc_RingAudioSource = NULL;
memset(&ht->ht_Voices[i].vc_SquareTempBuffer, 0, 0x80);
memset(&ht->ht_Voices[i].vc_ADSR, 0, sizeof(struct hvl_envelope));
memset(&ht->ht_Voices[i].vc_VoiceBuffer, 0, 0x281);
memset(&ht->ht_Voices[i].vc_RingVoiceBuffer, 0, 0x281);
}
for (i = 0; i < MAX_CHANNELS; i++) {
ht->ht_Voices[i].vc_WNRandom = 0x280;
ht->ht_Voices[i].vc_VoiceNum = i;
ht->ht_Voices[i].vc_TrackMasterVolume = 0x40;
ht->ht_Voices[i].vc_TrackOn = 1;
ht->ht_Voices[i].vc_MixSource = ht->ht_Voices[i].vc_VoiceBuffer;
}
2023-01-01 17:26:02 +00:00
}
BOOL hvl_InitSubsong(struct hvl_tune *ht, uint32 nr) {
uint32 PosNr, i;
if (nr > ht->ht_SubsongNr)
return FALSE;
ht->ht_SongNum = nr;
PosNr = 0;
if (nr)
PosNr = ht->ht_Subsongs[nr - 1];
ht->ht_PosNr = PosNr;
ht->ht_PosJump = 0;
ht->ht_PatternBreak = 0;
ht->ht_NoteNr = 0;
ht->ht_PosJumpNote = 0;
ht->ht_Tempo = 6;
ht->ht_StepWaitFrames = 0;
ht->ht_GetNewPosition = 1;
ht->ht_SongEndReached = 0;
ht->ht_PlayingTime = 0;
for (i = 0; i < MAX_CHANNELS; i += 4) {
ht->ht_Voices[i + 0].vc_Pan = ht->ht_defpanleft;
ht->ht_Voices[i + 0].vc_SetPan = ht->ht_defpanleft; // 1.4
ht->ht_Voices[i + 0].vc_PanMultLeft = panning_left[ht->ht_defpanleft];
ht->ht_Voices[i + 0].vc_PanMultRight = panning_right[ht->ht_defpanleft];
ht->ht_Voices[i + 1].vc_Pan = ht->ht_defpanright;
ht->ht_Voices[i + 1].vc_SetPan = ht->ht_defpanright; // 1.4
ht->ht_Voices[i + 1].vc_PanMultLeft = panning_left[ht->ht_defpanright];
ht->ht_Voices[i + 1].vc_PanMultRight = panning_right[ht->ht_defpanright];
ht->ht_Voices[i + 2].vc_Pan = ht->ht_defpanright;
ht->ht_Voices[i + 2].vc_SetPan = ht->ht_defpanright; // 1.4
ht->ht_Voices[i + 2].vc_PanMultLeft = panning_left[ht->ht_defpanright];
ht->ht_Voices[i + 2].vc_PanMultRight = panning_right[ht->ht_defpanright];
ht->ht_Voices[i + 3].vc_Pan = ht->ht_defpanleft;
ht->ht_Voices[i + 3].vc_SetPan = ht->ht_defpanleft; // 1.4
ht->ht_Voices[i + 3].vc_PanMultLeft = panning_left[ht->ht_defpanleft];
ht->ht_Voices[i + 3].vc_PanMultRight = panning_right[ht->ht_defpanleft];
}
hvl_reset_some_stuff(ht);
return TRUE;
2023-01-01 17:26:02 +00:00
}
void hvl_InitReplayer(void) { hvl_GenTables(); }
struct hvl_tune *hvl_load_ahx(const uint8 *buf, uint32 buflen, uint32 defstereo, uint32 freq) {
const uint8 *bptr;
const TEXT *nptr;
uint32 i, j, k, l, posn, insn, ssn, hs, trkn, trkl;
struct hvl_tune *ht;
struct hvl_plsentry *ple;
const int32 defgain[] = {71, 72, 76, 85, 100};
posn = ((buf[6] & 0x0f) << 8) | buf[7];
insn = buf[12];
ssn = buf[13];
trkl = buf[10];
trkn = buf[11];
hs = sizeof(struct hvl_tune);
hs += sizeof(struct hvl_position) * posn;
hs += sizeof(struct hvl_instrument) * (insn + 1);
hs += sizeof(uint16) * ssn;
// Calculate the size of all instrument PList buffers
bptr = &buf[14];
bptr += ssn * 2; // Skip past the subsong list
bptr += posn * 4 * 2; // Skip past the positions
bptr += trkn * trkl * 3;
if ((buf[6] & 0x80) == 0)
bptr += trkl * 3;
// *NOW* we can finally calculate PList space
for (i = 1; i <= insn; i++) {
hs += bptr[21] * sizeof(struct hvl_plsentry);
bptr += 22 + bptr[21] * 4;
}
ht = malloc(hs);
if (!ht) {
// printf("Out of memory!\n");
return NULL;
}
ht->ht_Frequency = freq;
ht->ht_FreqF = (float64)freq;
ht->ht_Positions = (struct hvl_position *)(&ht[1]);
ht->ht_Instruments = (struct hvl_instrument *)(&ht->ht_Positions[posn]);
ht->ht_Subsongs = (uint16 *)(&ht->ht_Instruments[(insn + 1)]);
ple = (struct hvl_plsentry *)(&ht->ht_Subsongs[ssn]);
ht->ht_WaveformTab[0] = &waves[WO_TRIANGLE_04];
ht->ht_WaveformTab[1] = &waves[WO_SAWTOOTH_04];
ht->ht_WaveformTab[3] = &waves[WO_WHITENOISE];
ht->ht_Channels = 4;
ht->ht_PositionNr = posn;
ht->ht_Restart = (buf[8] << 8) | buf[9];
ht->ht_SpeedMultiplier = ((buf[6] >> 5) & 3) + 1;
ht->ht_TrackLength = trkl;
ht->ht_TrackNr = trkn;
ht->ht_InstrumentNr = insn;
ht->ht_SubsongNr = ssn;
ht->ht_defstereo = defstereo;
ht->ht_defpanleft = stereopan_left[ht->ht_defstereo];
ht->ht_defpanright = stereopan_right[ht->ht_defstereo];
ht->ht_mixgain = (defgain[ht->ht_defstereo] * 256) / 100;
if (ht->ht_Restart >= ht->ht_PositionNr)
ht->ht_Restart = ht->ht_PositionNr - 1;
// Do some validation
if ((ht->ht_PositionNr > 1000) || (ht->ht_TrackLength > 64) || (ht->ht_InstrumentNr > 64)) {
// printf("%d,%d,%d\n", ht->ht_PositionNr, ht->ht_TrackLength, ht->ht_InstrumentNr);
free(ht);
// printf("Invalid file.\n");
return NULL;
}
strncpy(ht->ht_Name, (TEXT *)&buf[(buf[4] << 8) | buf[5]], 128);
nptr = (TEXT *)&buf[((buf[4] << 8) | buf[5]) + strlen(ht->ht_Name) + 1];
bptr = &buf[14];
// Subsongs
for (i = 0; i < ht->ht_SubsongNr; i++) {
ht->ht_Subsongs[i] = (bptr[0] << 8) | bptr[1];
if (ht->ht_Subsongs[i] >= ht->ht_PositionNr)
ht->ht_Subsongs[i] = 0;
bptr += 2;
}
// Position list
for (i = 0; i < ht->ht_PositionNr; i++) {
for (j = 0; j < 4; j++) {
ht->ht_Positions[i].pos_Track[j] = *bptr++;
ht->ht_Positions[i].pos_Transpose[j] = *(int8 *)bptr++;
}
}
// Tracks
for (i = 0; i <= ht->ht_TrackNr; i++) {
if (((buf[6] & 0x80) == 0x80) && (i == 0)) {
for (j = 0; j < ht->ht_TrackLength; j++) {
ht->ht_Tracks[i][j].stp_Note = 0;
ht->ht_Tracks[i][j].stp_Instrument = 0;
ht->ht_Tracks[i][j].stp_FX = 0;
ht->ht_Tracks[i][j].stp_FXParam = 0;
ht->ht_Tracks[i][j].stp_FXb = 0;
ht->ht_Tracks[i][j].stp_FXbParam = 0;
}
continue;
}
for (j = 0; j < ht->ht_TrackLength; j++) {
ht->ht_Tracks[i][j].stp_Note = (bptr[0] >> 2) & 0x3f;
ht->ht_Tracks[i][j].stp_Instrument = ((bptr[0] & 0x3) << 4) | (bptr[1] >> 4);
ht->ht_Tracks[i][j].stp_FX = bptr[1] & 0xf;
ht->ht_Tracks[i][j].stp_FXParam = bptr[2];
ht->ht_Tracks[i][j].stp_FXb = 0;
ht->ht_Tracks[i][j].stp_FXbParam = 0;
bptr += 3;
}
}
// Instruments
for (i = 1; i <= ht->ht_InstrumentNr; i++) {
if (nptr < (TEXT *)(buf + buflen)) {
strncpy(ht->ht_Instruments[i].ins_Name, nptr, 128);
nptr += strlen(nptr) + 1;
} else {
ht->ht_Instruments[i].ins_Name[0] = 0;
}
ht->ht_Instruments[i].ins_Volume = bptr[0];
ht->ht_Instruments[i].ins_FilterSpeed = ((bptr[1] >> 3) & 0x1f) | ((bptr[12] >> 2) & 0x20);
ht->ht_Instruments[i].ins_WaveLength = bptr[1] & 0x07;
ht->ht_Instruments[i].ins_Envelope.aFrames = bptr[2];
ht->ht_Instruments[i].ins_Envelope.aVolume = bptr[3];
ht->ht_Instruments[i].ins_Envelope.dFrames = bptr[4];
ht->ht_Instruments[i].ins_Envelope.dVolume = bptr[5];
ht->ht_Instruments[i].ins_Envelope.sFrames = bptr[6];
ht->ht_Instruments[i].ins_Envelope.rFrames = bptr[7];
ht->ht_Instruments[i].ins_Envelope.rVolume = bptr[8];
ht->ht_Instruments[i].ins_FilterLowerLimit = bptr[12] & 0x7f;
ht->ht_Instruments[i].ins_VibratoDelay = bptr[13];
ht->ht_Instruments[i].ins_HardCutReleaseFrames = (bptr[14] >> 4) & 0x07;
ht->ht_Instruments[i].ins_HardCutRelease = bptr[14] & 0x80 ? 1 : 0;
ht->ht_Instruments[i].ins_VibratoDepth = bptr[14] & 0x0f;
ht->ht_Instruments[i].ins_VibratoSpeed = bptr[15];
ht->ht_Instruments[i].ins_SquareLowerLimit = bptr[16];
ht->ht_Instruments[i].ins_SquareUpperLimit = bptr[17];
ht->ht_Instruments[i].ins_SquareSpeed = bptr[18];
ht->ht_Instruments[i].ins_FilterUpperLimit = bptr[19] & 0x3f;
ht->ht_Instruments[i].ins_PList.pls_Speed = bptr[20];
ht->ht_Instruments[i].ins_PList.pls_Length = bptr[21];
ht->ht_Instruments[i].ins_PList.pls_Entries = ple;
ple += bptr[21];
bptr += 22;
for (j = 0; j < ht->ht_Instruments[i].ins_PList.pls_Length; j++) {
k = (bptr[0] >> 5) & 7;
if (k == 6)
k = 12;
if (k == 7)
k = 15;
l = (bptr[0] >> 2) & 7;
if (l == 6)
l = 12;
if (l == 7)
l = 15;
ht->ht_Instruments[i].ins_PList.pls_Entries[j].ple_FX[1] = k;
ht->ht_Instruments[i].ins_PList.pls_Entries[j].ple_FX[0] = l;
ht->ht_Instruments[i].ins_PList.pls_Entries[j].ple_Waveform = ((bptr[0] << 1) & 6) | (bptr[1] >> 7);
ht->ht_Instruments[i].ins_PList.pls_Entries[j].ple_Fixed = (bptr[1] >> 6) & 1;
ht->ht_Instruments[i].ins_PList.pls_Entries[j].ple_Note = bptr[1] & 0x3f;
ht->ht_Instruments[i].ins_PList.pls_Entries[j].ple_FXParam[0] = bptr[2];
ht->ht_Instruments[i].ins_PList.pls_Entries[j].ple_FXParam[1] = bptr[3];
// 1.6: Strip "toggle filter" commands if the module is
// version 0 (pre-filters). This is what AHX also does.
if ((buf[3] == 0) && (l == 4) && ((bptr[2] & 0xf0) != 0))
ht->ht_Instruments[i].ins_PList.pls_Entries[j].ple_FXParam[0] &= 0x0f;
if ((buf[3] == 0) && (k == 4) && ((bptr[3] & 0xf0) != 0))
ht->ht_Instruments[i].ins_PList.pls_Entries[j].ple_FXParam[1] &= 0x0f; // 1.8
bptr += 4;
}
}
hvl_InitSubsong(ht, 0);
return ht;
2023-01-01 17:26:02 +00:00
}
struct hvl_tune *hvl_load_hvl(const uint8 *buf, uint32 buflen, uint32 defstereo, uint32 freq) {
const uint8 *bptr;
const TEXT *nptr;
uint32 i, j, posn, insn, ssn, chnn, hs, trkl, trkn;
struct hvl_tune *ht;
struct hvl_plsentry *ple;
posn = ((buf[6] & 0x0f) << 8) | buf[7];
insn = buf[12];
ssn = buf[13];
chnn = (buf[8] >> 2) + 4;
trkl = buf[10];
trkn = buf[11];
hs = sizeof(struct hvl_tune);
hs += sizeof(struct hvl_position) * posn;
hs += sizeof(struct hvl_instrument) * (insn + 1);
hs += sizeof(uint16) * ssn;
// Calculate the size of all instrument PList buffers
bptr = &buf[16];
bptr += ssn * 2; // Skip past the subsong list
bptr += posn * chnn * 2; // Skip past the positions
// Skip past the tracks
// 1.4: Fixed two really stupid bugs that cancelled each other
// out if the module had a blank first track (which is how
// come they were missed.
for (i = ((buf[6] & 0x80) == 0x80) ? 1 : 0; i <= trkn; i++)
for (j = 0; j < trkl; j++) {
if (bptr[0] == 0x3f) {
bptr++;
continue;
}
bptr += 5;
}
// *NOW* we can finally calculate PList space
for (i = 1; i <= insn; i++) {
hs += bptr[21] * sizeof(struct hvl_plsentry);
bptr += 22 + bptr[21] * 5;
}
ht = malloc(hs);
if (!ht) {
// printf("Out of memory!\n");
return NULL;
}
ht->ht_Version = buf[3]; // 1.5
ht->ht_Frequency = freq;
ht->ht_FreqF = (float64)freq;
ht->ht_Positions = (struct hvl_position *)(&ht[1]);
ht->ht_Instruments = (struct hvl_instrument *)(&ht->ht_Positions[posn]);
ht->ht_Subsongs = (uint16 *)(&ht->ht_Instruments[(insn + 1)]);
ple = (struct hvl_plsentry *)(&ht->ht_Subsongs[ssn]);
ht->ht_WaveformTab[0] = &waves[WO_TRIANGLE_04];
ht->ht_WaveformTab[1] = &waves[WO_SAWTOOTH_04];
ht->ht_WaveformTab[3] = &waves[WO_WHITENOISE];
ht->ht_PositionNr = posn;
ht->ht_Channels = (buf[8] >> 2) + 4;
ht->ht_Restart = ((buf[8] & 3) << 8) | buf[9];
ht->ht_SpeedMultiplier = ((buf[6] >> 5) & 3) + 1;
ht->ht_TrackLength = buf[10];
ht->ht_TrackNr = buf[11];
ht->ht_InstrumentNr = insn;
ht->ht_SubsongNr = ssn;
ht->ht_mixgain = (buf[14] << 8) / 100;
ht->ht_defstereo = buf[15];
ht->ht_defpanleft = stereopan_left[ht->ht_defstereo];
ht->ht_defpanright = stereopan_right[ht->ht_defstereo];
if (ht->ht_Restart >= ht->ht_PositionNr)
ht->ht_Restart = ht->ht_PositionNr - 1;
// Do some validation
if ((ht->ht_PositionNr > 1000) || (ht->ht_TrackLength > 64) || (ht->ht_InstrumentNr > 64)) {
// printf("%d,%d,%d\n", ht->ht_PositionNr, ht->ht_TrackLength, ht->ht_InstrumentNr);
free(ht);
// printf("Invalid file.\n");
return NULL;
}
strncpy(ht->ht_Name, (TEXT *)&buf[(buf[4] << 8) | buf[5]], 128);
nptr = (TEXT *)&buf[((buf[4] << 8) | buf[5]) + strlen(ht->ht_Name) + 1];
bptr = &buf[16];
// Subsongs
for (i = 0; i < ht->ht_SubsongNr; i++) {
ht->ht_Subsongs[i] = (bptr[0] << 8) | bptr[1];
bptr += 2;
}
// Position list
for (i = 0; i < ht->ht_PositionNr; i++) {
for (j = 0; j < ht->ht_Channels; j++) {
ht->ht_Positions[i].pos_Track[j] = *bptr++;
ht->ht_Positions[i].pos_Transpose[j] = *(int8 *)bptr++;
}
}
// Tracks
for (i = 0; i <= ht->ht_TrackNr; i++) {
if (((buf[6] & 0x80) == 0x80) && (i == 0)) {
for (j = 0; j < ht->ht_TrackLength; j++) {
ht->ht_Tracks[i][j].stp_Note = 0;
ht->ht_Tracks[i][j].stp_Instrument = 0;
ht->ht_Tracks[i][j].stp_FX = 0;
ht->ht_Tracks[i][j].stp_FXParam = 0;
ht->ht_Tracks[i][j].stp_FXb = 0;
ht->ht_Tracks[i][j].stp_FXbParam = 0;
}
continue;
}
for (j = 0; j < ht->ht_TrackLength; j++) {
if (bptr[0] == 0x3f) {
ht->ht_Tracks[i][j].stp_Note = 0;
ht->ht_Tracks[i][j].stp_Instrument = 0;
ht->ht_Tracks[i][j].stp_FX = 0;
ht->ht_Tracks[i][j].stp_FXParam = 0;
ht->ht_Tracks[i][j].stp_FXb = 0;
ht->ht_Tracks[i][j].stp_FXbParam = 0;
bptr++;
continue;
}
ht->ht_Tracks[i][j].stp_Note = bptr[0];
ht->ht_Tracks[i][j].stp_Instrument = bptr[1];
ht->ht_Tracks[i][j].stp_FX = bptr[2] >> 4;
ht->ht_Tracks[i][j].stp_FXParam = bptr[3];
ht->ht_Tracks[i][j].stp_FXb = bptr[2] & 0xf;
ht->ht_Tracks[i][j].stp_FXbParam = bptr[4];
bptr += 5;
}
}
// Instruments
for (i = 1; i <= ht->ht_InstrumentNr; i++) {
if (nptr < (TEXT *)(buf + buflen)) {
strncpy(ht->ht_Instruments[i].ins_Name, nptr, 128);
nptr += strlen(nptr) + 1;
} else {
ht->ht_Instruments[i].ins_Name[0] = 0;
}
ht->ht_Instruments[i].ins_Volume = bptr[0];
ht->ht_Instruments[i].ins_FilterSpeed = ((bptr[1] >> 3) & 0x1f) | ((bptr[12] >> 2) & 0x20);
ht->ht_Instruments[i].ins_WaveLength = bptr[1] & 0x07;
ht->ht_Instruments[i].ins_Envelope.aFrames = bptr[2];
ht->ht_Instruments[i].ins_Envelope.aVolume = bptr[3];
ht->ht_Instruments[i].ins_Envelope.dFrames = bptr[4];
ht->ht_Instruments[i].ins_Envelope.dVolume = bptr[5];
ht->ht_Instruments[i].ins_Envelope.sFrames = bptr[6];
ht->ht_Instruments[i].ins_Envelope.rFrames = bptr[7];
ht->ht_Instruments[i].ins_Envelope.rVolume = bptr[8];
ht->ht_Instruments[i].ins_FilterLowerLimit = bptr[12] & 0x7f;
ht->ht_Instruments[i].ins_VibratoDelay = bptr[13];
ht->ht_Instruments[i].ins_HardCutReleaseFrames = (bptr[14] >> 4) & 0x07;
ht->ht_Instruments[i].ins_HardCutRelease = bptr[14] & 0x80 ? 1 : 0;
ht->ht_Instruments[i].ins_VibratoDepth = bptr[14] & 0x0f;
ht->ht_Instruments[i].ins_VibratoSpeed = bptr[15];
ht->ht_Instruments[i].ins_SquareLowerLimit = bptr[16];
ht->ht_Instruments[i].ins_SquareUpperLimit = bptr[17];
ht->ht_Instruments[i].ins_SquareSpeed = bptr[18];
ht->ht_Instruments[i].ins_FilterUpperLimit = bptr[19] & 0x3f;
ht->ht_Instruments[i].ins_PList.pls_Speed = bptr[20];
ht->ht_Instruments[i].ins_PList.pls_Length = bptr[21];
ht->ht_Instruments[i].ins_PList.pls_Entries = ple;
ple += bptr[21];
bptr += 22;
for (j = 0; j < ht->ht_Instruments[i].ins_PList.pls_Length; j++) {
ht->ht_Instruments[i].ins_PList.pls_Entries[j].ple_FX[0] = bptr[0] & 0xf;
ht->ht_Instruments[i].ins_PList.pls_Entries[j].ple_FX[1] = (bptr[1] >> 3) & 0xf;
ht->ht_Instruments[i].ins_PList.pls_Entries[j].ple_Waveform = bptr[1] & 7;
ht->ht_Instruments[i].ins_PList.pls_Entries[j].ple_Fixed = (bptr[2] >> 6) & 1;
ht->ht_Instruments[i].ins_PList.pls_Entries[j].ple_Note = bptr[2] & 0x3f;
ht->ht_Instruments[i].ins_PList.pls_Entries[j].ple_FXParam[0] = bptr[3];
ht->ht_Instruments[i].ins_PList.pls_Entries[j].ple_FXParam[1] = bptr[4];
bptr += 5;
}
}
hvl_InitSubsong(ht, 0);
return ht;
2023-01-01 17:26:02 +00:00
}
struct hvl_tune *hvl_ParseTune(const uint8 *buf, uint32 buflen, uint32 freq, uint32 defstereo) {
struct hvl_tune *ht = NULL;
if ((buf[0] == 'T') && (buf[1] == 'H') && (buf[2] == 'X') && (buf[3] < 3)) {
ht = hvl_load_ahx(buf, buflen, defstereo, freq);
}
else if ((buf[0] == 'H') && (buf[1] == 'V') && (buf[2] == 'L') && (buf[3] < 2)) {
ht = hvl_load_hvl(buf, buflen, defstereo, freq);
} else {
// printf("Invalid file.\n");
}
return ht;
2023-01-01 17:26:02 +00:00
}
struct hvl_tune *hvl_LoadTune(const TEXT *name, uint32 freq, uint32 defstereo) {
struct hvl_tune *ht = NULL;
uint8 *buf;
uint32 buflen;
FILE *fh;
fh = fopen(name, "rb");
if (!fh) {
// printf("Can't open file\n");
return NULL;
}
fseek(fh, 0, SEEK_END);
buflen = ftell(fh);
fseek(fh, 0, SEEK_SET);
buf = malloc(buflen);
if (!buf) {
fclose(fh);
// printf("Out of memory!\n");
return NULL;
}
if (fread(buf, 1, buflen, fh) != buflen) {
fclose(fh);
free(buf);
// printf("Unable to read from file!\n");
return NULL;
}
fclose(fh);
ht = hvl_ParseTune(buf, buflen, freq, defstereo);
free(buf);
return ht;
2023-01-01 17:26:02 +00:00
}
void hvl_FreeTune(struct hvl_tune *ht) {
if (!ht)
return;
free(ht);
2023-01-01 17:26:02 +00:00
}
void hvl_process_stepfx_1(struct hvl_tune *ht, struct hvl_voice *voice, int32 FX, int32 FXParam) {
switch (FX) {
case 0x0: // Position Jump HI
if (((FXParam & 0x0f) > 0) && ((FXParam & 0x0f) <= 9))
ht->ht_PosJump = FXParam & 0xf;
break;
case 0x5: // Volume Slide + Tone Portamento
case 0xa: // Volume Slide
voice->vc_VolumeSlideDown = FXParam & 0x0f;
voice->vc_VolumeSlideUp = FXParam >> 4;
break;
case 0x7: // Panning
if (FXParam > 127)
FXParam -= 256;
voice->vc_Pan = (FXParam + 128);
voice->vc_SetPan = (FXParam + 128); // 1.4
voice->vc_PanMultLeft = panning_left[voice->vc_Pan];
voice->vc_PanMultRight = panning_right[voice->vc_Pan];
break;
case 0xb: // Position jump
ht->ht_PosJump = ht->ht_PosJump * 100 + (FXParam & 0x0f) + (FXParam >> 4) * 10;
ht->ht_PatternBreak = 1;
if (ht->ht_PosJump <= ht->ht_PosNr)
ht->ht_SongEndReached = 1;
break;
case 0xd: // Pattern break
ht->ht_PosJump = ht->ht_PosNr + 1;
ht->ht_PosJumpNote = (FXParam & 0x0f) + (FXParam >> 4) * 10;
ht->ht_PatternBreak = 1;
if (ht->ht_PosJumpNote > ht->ht_TrackLength)
ht->ht_PosJumpNote = 0;
break;
case 0xe: // Extended commands
switch (FXParam >> 4) {
case 0xc: // Note cut
if ((FXParam & 0x0f) < ht->ht_Tempo) {
voice->vc_NoteCutWait = FXParam & 0x0f;
if (voice->vc_NoteCutWait) {
voice->vc_NoteCutOn = 1;
voice->vc_HardCutRelease = 0;
}
}
break;
// 1.6: 0xd case removed
}
break;
case 0xf: // Speed
ht->ht_Tempo = FXParam;
if (FXParam == 0)
ht->ht_SongEndReached = 1;
break;
}
2023-01-01 17:26:02 +00:00
}
void hvl_process_stepfx_2(const struct hvl_tune *ht, struct hvl_voice *voice, int32 FX, int32 FXParam, int32 *Note) {
switch (FX) {
case 0x9: // Set squarewave offset
voice->vc_SquarePos = FXParam >> (5 - voice->vc_WaveLength);
// voice->vc_PlantSquare = 1;
voice->vc_IgnoreSquare = 1;
break;
case 0x3: // Tone portamento
if (FXParam != 0)
voice->vc_PeriodSlideSpeed = FXParam;
case 0x5: // Tone portamento + volume slide
if (*Note) {
int32 new, diff;
new = period_tab[*Note];
diff = period_tab[voice->vc_TrackPeriod];
diff -= new;
new = diff + voice->vc_PeriodSlidePeriod;
if (new)
voice->vc_PeriodSlideLimit = -diff;
}
voice->vc_PeriodSlideOn = 1;
voice->vc_PeriodSlideWithLimit = 1;
*Note = 0;
break;
}
2023-01-01 17:26:02 +00:00
}
void hvl_process_stepfx_3(struct hvl_tune *ht, struct hvl_voice *voice, int32 FX, int32 FXParam) {
int32 i;
switch (FX) {
case 0x01: // Portamento up (period slide down)
voice->vc_PeriodSlideSpeed = -FXParam;
voice->vc_PeriodSlideOn = 1;
voice->vc_PeriodSlideWithLimit = 0;
break;
case 0x02: // Portamento down
voice->vc_PeriodSlideSpeed = FXParam;
voice->vc_PeriodSlideOn = 1;
voice->vc_PeriodSlideWithLimit = 0;
break;
case 0x04: // Filter override
if ((FXParam == 0) || (FXParam == 0x40))
break;
if (FXParam < 0x40) {
voice->vc_IgnoreFilter = FXParam;
break;
}
if (FXParam > 0x7f)
break;
voice->vc_FilterPos = FXParam - 0x40;
break;
case 0x0c: // Volume
FXParam &= 0xff;
if (FXParam <= 0x40) {
voice->vc_NoteMaxVolume = FXParam;
break;
}
if ((FXParam -= 0x50) < 0)
break; // 1.6
if (FXParam <= 0x40) {
for (i = 0; i < ht->ht_Channels; i++)
ht->ht_Voices[i].vc_TrackMasterVolume = FXParam;
break;
}
if ((FXParam -= 0xa0 - 0x50) < 0)
break; // 1.6
if (FXParam <= 0x40)
voice->vc_TrackMasterVolume = FXParam;
break;
case 0xe: // Extended commands;
switch (FXParam >> 4) {
case 0x1: // Fineslide up
voice->vc_PeriodSlidePeriod -= (FXParam & 0x0f); // 1.8
voice->vc_PlantPeriod = 1;
break;
case 0x2: // Fineslide down
voice->vc_PeriodSlidePeriod += (FXParam & 0x0f); // 1.8
voice->vc_PlantPeriod = 1;
break;
case 0x4: // Vibrato control
voice->vc_VibratoDepth = FXParam & 0x0f;
break;
case 0x0a: // Fine volume up
voice->vc_NoteMaxVolume += FXParam & 0x0f;
if (voice->vc_NoteMaxVolume > 0x40)
voice->vc_NoteMaxVolume = 0x40;
break;
case 0x0b: // Fine volume down
voice->vc_NoteMaxVolume -= FXParam & 0x0f;
if (voice->vc_NoteMaxVolume < 0)
voice->vc_NoteMaxVolume = 0;
break;
case 0x0f: // Misc flags (1.5)
if (ht->ht_Version < 1)
break;
switch (FXParam & 0xf) {
case 1:
voice->vc_OverrideTranspose = voice->vc_Transpose;
break;
}
break;
}
break;
}
2023-01-01 17:26:02 +00:00
}
void hvl_process_step(struct hvl_tune *ht, struct hvl_voice *voice) {
int32 Note, Instr, donenotedel;
const struct hvl_step *Step;
if (voice->vc_TrackOn == 0)
return;
voice->vc_VolumeSlideUp = voice->vc_VolumeSlideDown = 0;
Step = &ht->ht_Tracks[ht->ht_Positions[ht->ht_PosNr].pos_Track[voice->vc_VoiceNum]][ht->ht_NoteNr];
Note = Step->stp_Note;
Instr = Step->stp_Instrument;
// --------- 1.6: from here --------------
donenotedel = 0;
// Do notedelay here
if (((Step->stp_FX & 0xf) == 0xe) && ((Step->stp_FXParam & 0xf0) == 0xd0)) {
if (voice->vc_NoteDelayOn) {
voice->vc_NoteDelayOn = 0;
donenotedel = 1;
} else {
if ((Step->stp_FXParam & 0x0f) < ht->ht_Tempo) {
voice->vc_NoteDelayWait = Step->stp_FXParam & 0x0f;
if (voice->vc_NoteDelayWait) {
voice->vc_NoteDelayOn = 1;
return;
}
}
}
}
if ((donenotedel == 0) && ((Step->stp_FXb & 0xf) == 0xe) && ((Step->stp_FXbParam & 0xf0) == 0xd0)) {
if (voice->vc_NoteDelayOn) {
voice->vc_NoteDelayOn = 0;
} else {
if ((Step->stp_FXbParam & 0x0f) < ht->ht_Tempo) {
voice->vc_NoteDelayWait = Step->stp_FXbParam & 0x0f;
if (voice->vc_NoteDelayWait) {
voice->vc_NoteDelayOn = 1;
return;
}
}
}
}
// --------- 1.6: to here --------------
if (Note)
voice->vc_OverrideTranspose = 1000; // 1.5
hvl_process_stepfx_1(ht, voice, Step->stp_FX & 0xf, Step->stp_FXParam);
hvl_process_stepfx_1(ht, voice, Step->stp_FXb & 0xf, Step->stp_FXbParam);
if ((Instr) && (Instr <= ht->ht_InstrumentNr)) {
struct hvl_instrument *Ins;
int16 SquareLower, SquareUpper, d6, d3, d4;
/* 1.4: Reset panning to last set position */
voice->vc_Pan = voice->vc_SetPan;
voice->vc_PanMultLeft = panning_left[voice->vc_Pan];
voice->vc_PanMultRight = panning_right[voice->vc_Pan];
voice->vc_PeriodSlideSpeed = voice->vc_PeriodSlidePeriod = voice->vc_PeriodSlideLimit = 0;
voice->vc_PerfSubVolume = 0x40;
voice->vc_ADSRVolume = 0;
voice->vc_Instrument = Ins = &ht->ht_Instruments[Instr];
voice->vc_SamplePos = 0;
voice->vc_ADSR.aFrames = Ins->ins_Envelope.aFrames;
voice->vc_ADSR.aVolume = voice->vc_ADSR.aFrames ? Ins->ins_Envelope.aVolume * 256 / voice->vc_ADSR.aFrames : Ins->ins_Envelope.aVolume * 256; // XXX
voice->vc_ADSR.dFrames = Ins->ins_Envelope.dFrames;
voice->vc_ADSR.dVolume = voice->vc_ADSR.dFrames ? (Ins->ins_Envelope.dVolume - Ins->ins_Envelope.aVolume) * 256 / voice->vc_ADSR.dFrames
: Ins->ins_Envelope.dVolume * 256; // XXX
voice->vc_ADSR.sFrames = Ins->ins_Envelope.sFrames;
voice->vc_ADSR.rFrames = Ins->ins_Envelope.rFrames;
voice->vc_ADSR.rVolume = voice->vc_ADSR.rFrames ? (Ins->ins_Envelope.rVolume - Ins->ins_Envelope.dVolume) * 256 / voice->vc_ADSR.rFrames
: Ins->ins_Envelope.rVolume * 256; // XXX
voice->vc_WaveLength = Ins->ins_WaveLength;
voice->vc_NoteMaxVolume = Ins->ins_Volume;
voice->vc_VibratoCurrent = 0;
voice->vc_VibratoDelay = Ins->ins_VibratoDelay;
voice->vc_VibratoDepth = Ins->ins_VibratoDepth;
voice->vc_VibratoSpeed = Ins->ins_VibratoSpeed;
voice->vc_VibratoPeriod = 0;
voice->vc_HardCutRelease = Ins->ins_HardCutRelease;
voice->vc_HardCut = Ins->ins_HardCutReleaseFrames;
voice->vc_IgnoreSquare = voice->vc_SquareSlidingIn = 0;
voice->vc_SquareWait = voice->vc_SquareOn = 0;
SquareLower = Ins->ins_SquareLowerLimit >> (5 - voice->vc_WaveLength);
SquareUpper = Ins->ins_SquareUpperLimit >> (5 - voice->vc_WaveLength);
if (SquareUpper < SquareLower) {
int16 t = SquareUpper;
SquareUpper = SquareLower;
SquareLower = t;
}
voice->vc_SquareUpperLimit = SquareUpper;
voice->vc_SquareLowerLimit = SquareLower;
voice->vc_IgnoreFilter = voice->vc_FilterWait = voice->vc_FilterOn = 0;
voice->vc_FilterSlidingIn = 0;
d6 = Ins->ins_FilterSpeed;
d3 = Ins->ins_FilterLowerLimit;
d4 = Ins->ins_FilterUpperLimit;
if (d3 & 0x80)
d6 |= 0x20;
if (d4 & 0x80)
d6 |= 0x40;
voice->vc_FilterSpeed = d6;
d3 &= ~0x80;
d4 &= ~0x80;
if (d3 > d4) {
int16 t = d3;
d3 = d4;
d4 = t;
}
voice->vc_FilterUpperLimit = d4;
voice->vc_FilterLowerLimit = d3;
voice->vc_FilterPos = 32;
voice->vc_PerfWait = voice->vc_PerfCurrent = 0;
voice->vc_PerfSpeed = Ins->ins_PList.pls_Speed;
voice->vc_PerfList = &voice->vc_Instrument->ins_PList;
voice->vc_RingMixSource = NULL; // No ring modulation
voice->vc_RingSamplePos = 0;
voice->vc_RingPlantPeriod = 0;
voice->vc_RingNewWaveform = 0;
}
voice->vc_PeriodSlideOn = 0;
2023-01-01 17:26:02 +00:00
hvl_process_stepfx_2(ht, voice, Step->stp_FX & 0xf, Step->stp_FXParam, &Note);
hvl_process_stepfx_2(ht, voice, Step->stp_FXb & 0xf, Step->stp_FXbParam, &Note);
if (Note) {
voice->vc_TrackPeriod = Note;
voice->vc_PlantPeriod = 1;
}
hvl_process_stepfx_3(ht, voice, Step->stp_FX & 0xf, Step->stp_FXParam);
hvl_process_stepfx_3(ht, voice, Step->stp_FXb & 0xf, Step->stp_FXbParam);
2023-01-01 17:26:02 +00:00
}
void hvl_plist_command_parse(const struct hvl_tune *ht, struct hvl_voice *voice, int32 FX, int32 FXParam) {
switch (FX) {
case 0:
if ((FXParam > 0) && (FXParam < 0x40)) {
if (voice->vc_IgnoreFilter) {
voice->vc_FilterPos = voice->vc_IgnoreFilter;
voice->vc_IgnoreFilter = 0;
} else {
voice->vc_FilterPos = FXParam;
}
voice->vc_NewWaveform = 1;
}
break;
case 1:
voice->vc_PeriodPerfSlideSpeed = FXParam;
voice->vc_PeriodPerfSlideOn = 1;
break;
case 2:
voice->vc_PeriodPerfSlideSpeed = -FXParam;
voice->vc_PeriodPerfSlideOn = 1;
break;
case 3:
if (voice->vc_IgnoreSquare == 0)
voice->vc_SquarePos = FXParam >> (5 - voice->vc_WaveLength);
else
voice->vc_IgnoreSquare = 0;
break;
case 4:
if (FXParam == 0) {
voice->vc_SquareInit = (voice->vc_SquareOn ^= 1);
voice->vc_SquareSign = 1;
} else {
if (FXParam & 0x0f) {
voice->vc_SquareInit = (voice->vc_SquareOn ^= 1);
voice->vc_SquareSign = 1;
if ((FXParam & 0x0f) == 0x0f)
voice->vc_SquareSign = -1;
}
if (FXParam & 0xf0) {
voice->vc_FilterInit = (voice->vc_FilterOn ^= 1);
voice->vc_FilterSign = 1;
if ((FXParam & 0xf0) == 0xf0)
voice->vc_FilterSign = -1;
}
}
break;
case 5:
voice->vc_PerfCurrent = FXParam;
break;
case 7:
// Ring modulate with triangle
if ((FXParam >= 1) && (FXParam <= 0x3C)) {
voice->vc_RingBasePeriod = FXParam;
voice->vc_RingFixedPeriod = 1;
} else if ((FXParam >= 0x81) && (FXParam <= 0xBC)) {
voice->vc_RingBasePeriod = FXParam - 0x80;
voice->vc_RingFixedPeriod = 0;
} else {
voice->vc_RingBasePeriod = 0;
voice->vc_RingFixedPeriod = 0;
voice->vc_RingNewWaveform = 0;
voice->vc_RingAudioSource = NULL; // turn it off
voice->vc_RingMixSource = NULL;
break;
}
voice->vc_RingWaveform = 0;
voice->vc_RingNewWaveform = 1;
voice->vc_RingPlantPeriod = 1;
break;
case 8: // Ring modulate with sawtooth
if ((FXParam >= 1) && (FXParam <= 0x3C)) {
voice->vc_RingBasePeriod = FXParam;
voice->vc_RingFixedPeriod = 1;
} else if ((FXParam >= 0x81) && (FXParam <= 0xBC)) {
voice->vc_RingBasePeriod = FXParam - 0x80;
voice->vc_RingFixedPeriod = 0;
} else {
voice->vc_RingBasePeriod = 0;
voice->vc_RingFixedPeriod = 0;
voice->vc_RingNewWaveform = 0;
voice->vc_RingAudioSource = NULL;
voice->vc_RingMixSource = NULL;
break;
}
voice->vc_RingWaveform = 1;
voice->vc_RingNewWaveform = 1;
voice->vc_RingPlantPeriod = 1;
break;
/* New in HivelyTracker 1.4 */
case 9:
if (FXParam > 127)
FXParam -= 256;
voice->vc_Pan = (FXParam + 128);
voice->vc_PanMultLeft = panning_left[voice->vc_Pan];
voice->vc_PanMultRight = panning_right[voice->vc_Pan];
break;
case 12:
if (FXParam <= 0x40) {
voice->vc_NoteMaxVolume = FXParam;
break;
}
if ((FXParam -= 0x50) < 0)
break;
if (FXParam <= 0x40) {
voice->vc_PerfSubVolume = FXParam;
break;
}
if ((FXParam -= 0xa0 - 0x50) < 0)
break;
if (FXParam <= 0x40)
voice->vc_TrackMasterVolume = FXParam;
break;
case 15:
voice->vc_PerfSpeed = voice->vc_PerfWait = FXParam;
break;
}
2023-01-01 17:26:02 +00:00
}
void hvl_process_frame(struct hvl_tune *ht, struct hvl_voice *voice) {
static const uint8 Offsets[] = {0x00, 0x04, 0x04 + 0x08, 0x04 + 0x08 + 0x10, 0x04 + 0x08 + 0x10 + 0x20, 0x04 + 0x08 + 0x10 + 0x20 + 0x40};
if (voice->vc_TrackOn == 0)
return;
if (voice->vc_NoteDelayOn) {
if (voice->vc_NoteDelayWait <= 0)
hvl_process_step(ht, voice);
else
voice->vc_NoteDelayWait--;
}
if (voice->vc_HardCut) {
int32 nextinst;
if (ht->ht_NoteNr + 1 < ht->ht_TrackLength)
nextinst = ht->ht_Tracks[voice->vc_Track][ht->ht_NoteNr + 1].stp_Instrument;
else
nextinst = ht->ht_Tracks[voice->vc_NextTrack][0].stp_Instrument;
if (nextinst) {
int32 d1;
d1 = ht->ht_Tempo - voice->vc_HardCut;
if (d1 < 0)
d1 = 0;
if (!voice->vc_NoteCutOn) {
voice->vc_NoteCutOn = 1;
voice->vc_NoteCutWait = d1;
voice->vc_HardCutReleaseF = -(d1 - ht->ht_Tempo);
} else {
voice->vc_HardCut = 0;
}
}
}
if (voice->vc_NoteCutOn) {
if (voice->vc_NoteCutWait <= 0) {
voice->vc_NoteCutOn = 0;
if (voice->vc_HardCutRelease) {
voice->vc_ADSR.rFrames = voice->vc_HardCutReleaseF;
voice->vc_ADSR.rVolume = 0;
if (voice->vc_ADSR.rFrames > 0)
voice->vc_ADSR.rVolume = -(voice->vc_ADSRVolume - (voice->vc_Instrument->ins_Envelope.rVolume << 8)) / voice->vc_ADSR.rFrames;
voice->vc_ADSR.aFrames = voice->vc_ADSR.dFrames = voice->vc_ADSR.sFrames = 0;
} else {
voice->vc_NoteMaxVolume = 0;
}
} else {
voice->vc_NoteCutWait--;
}
}
// ADSR envelope
if (voice->vc_ADSR.aFrames) {
voice->vc_ADSRVolume += voice->vc_ADSR.aVolume;
if (--voice->vc_ADSR.aFrames <= 0)
voice->vc_ADSRVolume = voice->vc_Instrument->ins_Envelope.aVolume << 8;
} else if (voice->vc_ADSR.dFrames) {
voice->vc_ADSRVolume += voice->vc_ADSR.dVolume;
if (--voice->vc_ADSR.dFrames <= 0)
voice->vc_ADSRVolume = voice->vc_Instrument->ins_Envelope.dVolume << 8;
} else if (voice->vc_ADSR.sFrames) {
voice->vc_ADSR.sFrames--;
} else if (voice->vc_ADSR.rFrames) {
voice->vc_ADSRVolume += voice->vc_ADSR.rVolume;
if (--voice->vc_ADSR.rFrames <= 0)
voice->vc_ADSRVolume = voice->vc_Instrument->ins_Envelope.rVolume << 8;
}
// VolumeSlide
voice->vc_NoteMaxVolume = voice->vc_NoteMaxVolume + voice->vc_VolumeSlideUp - voice->vc_VolumeSlideDown;
if (voice->vc_NoteMaxVolume < 0)
voice->vc_NoteMaxVolume = 0;
else if (voice->vc_NoteMaxVolume > 0x40)
voice->vc_NoteMaxVolume = 0x40;
// Portamento
if (voice->vc_PeriodSlideOn) {
if (voice->vc_PeriodSlideWithLimit) {
int32 d0, d2;
d0 = voice->vc_PeriodSlidePeriod - voice->vc_PeriodSlideLimit;
d2 = voice->vc_PeriodSlideSpeed;
if (d0 > 0)
d2 = -d2;
if (d0) {
int32 d3;
d3 = (d0 + d2) ^ d0;
if (d3 >= 0)
d0 = voice->vc_PeriodSlidePeriod + d2;
else
d0 = voice->vc_PeriodSlideLimit;
voice->vc_PeriodSlidePeriod = d0;
voice->vc_PlantPeriod = 1;
}
} else {
voice->vc_PeriodSlidePeriod += voice->vc_PeriodSlideSpeed;
voice->vc_PlantPeriod = 1;
}
}
// Vibrato
if (voice->vc_VibratoDepth) {
if (voice->vc_VibratoDelay <= 0) {
voice->vc_VibratoPeriod = (vib_tab[voice->vc_VibratoCurrent] * voice->vc_VibratoDepth) >> 7;
voice->vc_PlantPeriod = 1;
voice->vc_VibratoCurrent = (voice->vc_VibratoCurrent + voice->vc_VibratoSpeed) & 0x3f;
} else {
voice->vc_VibratoDelay--;
}
}
// PList
if (voice->vc_PerfList != 0) {
if (voice->vc_Instrument && voice->vc_PerfCurrent < voice->vc_Instrument->ins_PList.pls_Length) {
int signedOverflow = (voice->vc_PerfWait == 128);
voice->vc_PerfWait--;
if (signedOverflow || (int8)voice->vc_PerfWait <= 0) {
uint32 i;
int32 cur;
cur = voice->vc_PerfCurrent++;
voice->vc_PerfWait = voice->vc_PerfSpeed;
if (voice->vc_PerfList->pls_Entries[cur].ple_Waveform) {
voice->vc_Waveform = voice->vc_PerfList->pls_Entries[cur].ple_Waveform - 1;
voice->vc_NewWaveform = 1;
voice->vc_PeriodPerfSlideSpeed = voice->vc_PeriodPerfSlidePeriod = 0;
}
// Holdwave
voice->vc_PeriodPerfSlideOn = 0;
for (i = 0; i < 2; i++)
hvl_plist_command_parse(ht, voice, voice->vc_PerfList->pls_Entries[cur].ple_FX[i] & 0xff,
voice->vc_PerfList->pls_Entries[cur].ple_FXParam[i] & 0xff);
// GetNote
if (voice->vc_PerfList->pls_Entries[cur].ple_Note) {
voice->vc_InstrPeriod = voice->vc_PerfList->pls_Entries[cur].ple_Note;
voice->vc_PlantPeriod = 1;
voice->vc_FixedNote = voice->vc_PerfList->pls_Entries[cur].ple_Fixed;
}
}
} else {
if (voice->vc_PerfWait)
voice->vc_PerfWait--;
else
voice->vc_PeriodPerfSlideSpeed = 0;
}
}
// PerfPortamento
if (voice->vc_PeriodPerfSlideOn) {
voice->vc_PeriodPerfSlidePeriod -= voice->vc_PeriodPerfSlideSpeed;
if (voice->vc_PeriodPerfSlidePeriod)
voice->vc_PlantPeriod = 1;
}
if (voice->vc_Waveform == 3 - 1 && voice->vc_SquareOn) {
if (--voice->vc_SquareWait <= 0) {
int32 d1, d2, d3;
d1 = voice->vc_SquareLowerLimit;
d2 = voice->vc_SquareUpperLimit;
d3 = voice->vc_SquarePos;
if (voice->vc_SquareInit) {
voice->vc_SquareInit = 0;
if (d3 <= d1) {
voice->vc_SquareSlidingIn = 1;
voice->vc_SquareSign = 1;
} else if (d3 >= d2) {
voice->vc_SquareSlidingIn = 1;
voice->vc_SquareSign = -1;
}
}
// NoSquareInit
if (d1 == d3 || d2 == d3) {
if (voice->vc_SquareSlidingIn)
voice->vc_SquareSlidingIn = 0;
else
voice->vc_SquareSign = -voice->vc_SquareSign;
}
d3 += voice->vc_SquareSign;
voice->vc_SquarePos = d3;
voice->vc_PlantSquare = 1;
voice->vc_SquareWait = voice->vc_Instrument->ins_SquareSpeed;
}
}
if (voice->vc_FilterOn && --voice->vc_FilterWait <= 0) {
uint32 i, FMax;
int32 d1, d2, d3;
d1 = voice->vc_FilterLowerLimit;
d2 = voice->vc_FilterUpperLimit;
d3 = voice->vc_FilterPos;
if (voice->vc_FilterInit) {
voice->vc_FilterInit = 0;
if (d3 <= d1) {
voice->vc_FilterSlidingIn = 1;
voice->vc_FilterSign = 1;
} else if (d3 >= d2) {
voice->vc_FilterSlidingIn = 1;
voice->vc_FilterSign = -1;
}
}
// NoFilterInit
FMax = (voice->vc_FilterSpeed < 4) ? (5 - voice->vc_FilterSpeed) : 1;
for (i = 0; i < FMax; i++) {
if ((d1 == d3) || (d2 == d3)) {
if (voice->vc_FilterSlidingIn)
voice->vc_FilterSlidingIn = 0;
else
voice->vc_FilterSign = -voice->vc_FilterSign;
}
d3 += voice->vc_FilterSign;
}
if (d3 < 1)
d3 = 1;
if (d3 > 63)
d3 = 63;
voice->vc_FilterPos = d3;
voice->vc_NewWaveform = 1;
voice->vc_FilterWait = voice->vc_FilterSpeed - 3;
if (voice->vc_FilterWait < 1)
voice->vc_FilterWait = 1;
}
if (voice->vc_Waveform == 3 - 1 || voice->vc_PlantSquare) {
// CalcSquare
uint32 i;
int32 Delta;
const int8 *SquarePtr;
int32 X;
SquarePtr = &waves[WO_SQUARES + (voice->vc_FilterPos - 0x20) * (0xfc + 0xfc + 0x80 * 0x1f + 0x80 + 0x280 * 3)];
X = voice->vc_SquarePos << (5 - voice->vc_WaveLength);
if (X > 0x20) {
X = 0x40 - X;
voice->vc_SquareReverse = 1;
}
// OkDownSquare
if (X > 0)
SquarePtr += (X - 1) << 7;
Delta = 32 >> voice->vc_WaveLength;
ht->ht_WaveformTab[2] = voice->vc_SquareTempBuffer;
for (i = 0; i < (1 << voice->vc_WaveLength) * 4; i++) {
voice->vc_SquareTempBuffer[i] = *SquarePtr;
SquarePtr += Delta;
}
voice->vc_NewWaveform = 1;
voice->vc_Waveform = 3 - 1;
voice->vc_PlantSquare = 0;
}
if (voice->vc_Waveform == 4 - 1)
voice->vc_NewWaveform = 1;
if (voice->vc_RingNewWaveform) {
const int8 *rasrc;
if (voice->vc_RingWaveform > 1)
voice->vc_RingWaveform = 1;
rasrc = ht->ht_WaveformTab[voice->vc_RingWaveform];
rasrc += Offsets[voice->vc_WaveLength];
voice->vc_RingAudioSource = rasrc;
}
if (voice->vc_NewWaveform) {
const int8 *AudioSource;
AudioSource = ht->ht_WaveformTab[voice->vc_Waveform];
if (voice->vc_Waveform != 3 - 1)
AudioSource += (voice->vc_FilterPos - 0x20) * (0xfc + 0xfc + 0x80 * 0x1f + 0x80 + 0x280 * 3);
if (voice->vc_Waveform < 3 - 1) {
// GetWLWaveformlor2
AudioSource += Offsets[voice->vc_WaveLength];
}
if (voice->vc_Waveform == 4 - 1) {
// AddRandomMoving
AudioSource += (voice->vc_WNRandom & (2 * 0x280 - 1)) & ~1;
// GoOnRandom
voice->vc_WNRandom += 2239384;
voice->vc_WNRandom = ((((voice->vc_WNRandom >> 8) | (voice->vc_WNRandom << 24)) + 782323) ^ 75) - 6735;
}
2023-01-01 17:26:02 +00:00
voice->vc_AudioSource = AudioSource;
}
2023-01-01 17:26:02 +00:00
// Ring modulation period calculation
if (voice->vc_RingAudioSource) {
voice->vc_RingAudioPeriod = voice->vc_RingBasePeriod;
2023-01-01 17:26:02 +00:00
if (!(voice->vc_RingFixedPeriod)) {
if (voice->vc_OverrideTranspose != 1000) // 1.5
voice->vc_RingAudioPeriod += voice->vc_OverrideTranspose + voice->vc_TrackPeriod - 1;
else
voice->vc_RingAudioPeriod += voice->vc_Transpose + voice->vc_TrackPeriod - 1;
}
2023-01-01 17:26:02 +00:00
if (voice->vc_RingAudioPeriod > 5 * 12)
voice->vc_RingAudioPeriod = 5 * 12;
2023-01-01 17:26:02 +00:00
if (voice->vc_RingAudioPeriod < 0)
voice->vc_RingAudioPeriod = 0;
2023-01-01 17:26:02 +00:00
voice->vc_RingAudioPeriod = period_tab[voice->vc_RingAudioPeriod];
2023-01-01 17:26:02 +00:00
if (!(voice->vc_RingFixedPeriod))
voice->vc_RingAudioPeriod += voice->vc_PeriodSlidePeriod;
2023-01-01 17:26:02 +00:00
voice->vc_RingAudioPeriod += voice->vc_PeriodPerfSlidePeriod + voice->vc_VibratoPeriod;
2023-01-01 17:26:02 +00:00
if (voice->vc_RingAudioPeriod > 0x0d60)
voice->vc_RingAudioPeriod = 0x0d60;
2023-01-01 17:26:02 +00:00
if (voice->vc_RingAudioPeriod < 0x0071)
voice->vc_RingAudioPeriod = 0x0071;
}
2023-01-01 17:26:02 +00:00
// Normal period calculation
voice->vc_AudioPeriod = voice->vc_InstrPeriod;
2023-01-01 17:26:02 +00:00
if (!(voice->vc_FixedNote)) {
if (voice->vc_OverrideTranspose != 1000) // 1.5
voice->vc_AudioPeriod += voice->vc_OverrideTranspose + voice->vc_TrackPeriod - 1;
else
voice->vc_AudioPeriod += voice->vc_Transpose + voice->vc_TrackPeriod - 1;
}
2023-01-01 17:26:02 +00:00
if (voice->vc_AudioPeriod > 5 * 12)
voice->vc_AudioPeriod = 5 * 12;
2023-01-01 17:26:02 +00:00
if (voice->vc_AudioPeriod < 0)
voice->vc_AudioPeriod = 0;
2023-01-01 17:26:02 +00:00
voice->vc_AudioPeriod = period_tab[voice->vc_AudioPeriod];
2023-01-01 17:26:02 +00:00
if (!(voice->vc_FixedNote))
voice->vc_AudioPeriod += voice->vc_PeriodSlidePeriod;
2023-01-01 17:26:02 +00:00
voice->vc_AudioPeriod += voice->vc_PeriodPerfSlidePeriod + voice->vc_VibratoPeriod;
if (voice->vc_AudioPeriod > 0x0d60)
voice->vc_AudioPeriod = 0x0d60;
if (voice->vc_AudioPeriod < 0x0071)
voice->vc_AudioPeriod = 0x0071;
voice->vc_AudioVolume =
(((((((voice->vc_ADSRVolume >> 8) * voice->vc_NoteMaxVolume) >> 6) * voice->vc_PerfSubVolume) >> 6) * voice->vc_TrackMasterVolume) >> 6);
2023-01-01 17:26:02 +00:00
}
void hvl_set_audio(struct hvl_voice *voice, float64 freqf) {
if (voice->vc_TrackOn == 0) {
voice->vc_VoiceVolume = 0;
return;
}
voice->vc_VoiceVolume = voice->vc_AudioVolume;
if (voice->vc_PlantPeriod) {
float64 freq2;
uint32 delta;
voice->vc_PlantPeriod = 0;
voice->vc_VoicePeriod = voice->vc_AudioPeriod;
freq2 = Period2Freq(voice->vc_AudioPeriod);
delta = (uint32)(freq2 / freqf);
if (delta > (0x280 << 16))
delta -= (0x280 << 16);
if (delta == 0)
delta = 1;
voice->vc_Delta = delta;
}
if (voice->vc_NewWaveform) {
const int8 *src;
src = voice->vc_AudioSource;
if (voice->vc_Waveform == 4 - 1) {
memcpy(&voice->vc_VoiceBuffer[0], src, 0x280);
} else {
uint32 i, WaveLoops;
WaveLoops = (1 << (5 - voice->vc_WaveLength)) * 5;
for (i = 0; i < WaveLoops; i++)
memcpy(&voice->vc_VoiceBuffer[i * 4 * (1 << voice->vc_WaveLength)], src, 4 * (1 << voice->vc_WaveLength));
}
voice->vc_VoiceBuffer[0x280] = voice->vc_VoiceBuffer[0];
voice->vc_MixSource = voice->vc_VoiceBuffer;
}
/* Ring Modulation */
if (voice->vc_RingPlantPeriod) {
float64 freq2;
uint32 delta;
voice->vc_RingPlantPeriod = 0;
freq2 = Period2Freq(voice->vc_RingAudioPeriod);
delta = (uint32)(freq2 / freqf);
if (delta > (0x280 << 16))
delta -= (0x280 << 16);
if (delta == 0)
delta = 1;
voice->vc_RingDelta = delta;
}
if (voice->vc_RingNewWaveform) {
const int8 *src;
uint32 i, WaveLoops;
src = voice->vc_RingAudioSource;
WaveLoops = (1 << (5 - voice->vc_WaveLength)) * 5;
for (i = 0; i < WaveLoops; i++)
memcpy(&voice->vc_RingVoiceBuffer[i * 4 * (1 << voice->vc_WaveLength)], src, 4 * (1 << voice->vc_WaveLength));
voice->vc_RingVoiceBuffer[0x280] = voice->vc_RingVoiceBuffer[0];
voice->vc_RingMixSource = voice->vc_RingVoiceBuffer;
}
2023-01-01 17:26:02 +00:00
}
void hvl_play_irq(struct hvl_tune *ht) {
uint32 i;
if (ht->ht_StepWaitFrames == 0) {
if (ht->ht_GetNewPosition) {
int32 nextpos = (ht->ht_PosNr + 1 == ht->ht_PositionNr) ? 0 : (ht->ht_PosNr + 1);
for (i = 0; i < ht->ht_Channels; i++) {
ht->ht_Voices[i].vc_Track = ht->ht_Positions[ht->ht_PosNr].pos_Track[i];
ht->ht_Voices[i].vc_Transpose = ht->ht_Positions[ht->ht_PosNr].pos_Transpose[i];
ht->ht_Voices[i].vc_NextTrack = ht->ht_Positions[nextpos].pos_Track[i];
ht->ht_Voices[i].vc_NextTranspose = ht->ht_Positions[nextpos].pos_Transpose[i];
}
ht->ht_GetNewPosition = 0;
}
for (i = 0; i < ht->ht_Channels; i++)
hvl_process_step(ht, &ht->ht_Voices[i]);
ht->ht_StepWaitFrames = ht->ht_Tempo;
}
for (i = 0; i < ht->ht_Channels; i++)
hvl_process_frame(ht, &ht->ht_Voices[i]);
ht->ht_PlayingTime++;
if (--ht->ht_StepWaitFrames == 0) {
if (!ht->ht_PatternBreak) {
ht->ht_NoteNr++;
if (ht->ht_NoteNr >= ht->ht_TrackLength) {
ht->ht_PosJump = ht->ht_PosNr + 1;
ht->ht_PosJumpNote = 0;
ht->ht_PatternBreak = 1;
}
}
if (ht->ht_PatternBreak) {
ht->ht_PatternBreak = 0;
ht->ht_PosNr = ht->ht_PosJump;
ht->ht_NoteNr = ht->ht_PosJumpNote;
if (ht->ht_PosNr == ht->ht_PositionNr) {
ht->ht_SongEndReached = 1;
ht->ht_PosNr = ht->ht_Restart;
}
ht->ht_PosJumpNote = 0;
ht->ht_PosJump = 0;
ht->ht_GetNewPosition = 1;
}
}
for (i = 0; i < ht->ht_Channels; i++)
hvl_set_audio(&ht->ht_Voices[i], ht->ht_Frequency);
2023-01-01 17:26:02 +00:00
}
void hvl_mixchunk(struct hvl_tune *ht, uint32 samples, int8 *buf1, int8 *buf2, int32 bufmod) {
const int8 *src[MAX_CHANNELS];
const int8 *rsrc[MAX_CHANNELS];
uint32 delta[MAX_CHANNELS];
uint32 rdelta[MAX_CHANNELS];
int32 vol[MAX_CHANNELS];
uint32 pos[MAX_CHANNELS];
uint32 rpos[MAX_CHANNELS];
uint32 cnt;
int32 panl[MAX_CHANNELS];
int32 panr[MAX_CHANNELS];
// uint32 vu[MAX_CHANNELS];
int32 a = 0, b = 0, j;
uint32 i, chans, loops;
chans = ht->ht_Channels;
for (i = 0; i < chans; i++) {
delta[i] = ht->ht_Voices[i].vc_Delta;
vol[i] = ht->ht_Voices[i].vc_VoiceVolume;
pos[i] = ht->ht_Voices[i].vc_SamplePos;
src[i] = ht->ht_Voices[i].vc_MixSource;
panl[i] = ht->ht_Voices[i].vc_PanMultLeft;
panr[i] = ht->ht_Voices[i].vc_PanMultRight;
/* Ring Modulation */
rdelta[i] = ht->ht_Voices[i].vc_RingDelta;
rpos[i] = ht->ht_Voices[i].vc_RingSamplePos;
rsrc[i] = ht->ht_Voices[i].vc_RingMixSource;
// vu[i] = 0;
}
do {
loops = samples;
for (i = 0; i < chans; i++) {
if (pos[i] >= (0x280 << 16))
pos[i] -= 0x280 << 16;
cnt = ((0x280 << 16) - pos[i] - 1) / delta[i] + 1;
if (cnt < loops)
loops = cnt;
if (rsrc[i]) {
if (rpos[i] >= (0x280 << 16))
rpos[i] -= 0x280 << 16;
cnt = ((0x280 << 16) - rpos[i] - 1) / rdelta[i] + 1;
if (cnt < loops)
loops = cnt;
}
}
samples -= loops;
// Inner loop
do {
a = 0;
b = 0;
for (i = 0; i < chans; i++) {
if (rsrc[i]) {
/* Ring Modulation */
j = ((src[i][pos[i] >> 16] * rsrc[i][rpos[i] >> 16]) >> 7) * vol[i];
rpos[i] += rdelta[i];
} else {
j = src[i][pos[i] >> 16] * vol[i];
}
// if( abs( j ) > vu[i] ) vu[i] = abs( j );
a += (j * panl[i]) >> 7;
b += (j * panr[i]) >> 7;
pos[i] += delta[i];
}
a = (a * ht->ht_mixgain) >> 8;
b = (b * ht->ht_mixgain) >> 8;
if (a < -0x8000)
a = -0x8000;
if (a > 0x7fff)
a = 0x7fff;
if (b < -0x8000)
b = -0x8000;
if (b > 0x7fff)
b = 0x7fff;
*(int16 *)buf1 = a;
*(int16 *)buf2 = b;
loops--;
buf1 += bufmod;
buf2 += bufmod;
} while (loops > 0);
} while (samples > 0);
for (i = 0; i < chans; i++) {
ht->ht_Voices[i].vc_SamplePos = pos[i];
ht->ht_Voices[i].vc_RingSamplePos = rpos[i];
// ht->ht_Voices[i].vc_VUMeter = vu[i];
}
2023-01-01 17:26:02 +00:00
}
void hvl_DecodeFrame(struct hvl_tune *ht, int8 *buf1, int8 *buf2, int32 bufmod) {
uint32 samples, loops;
samples = ht->ht_Frequency / 50 / ht->ht_SpeedMultiplier;
loops = ht->ht_SpeedMultiplier;
do {
hvl_play_irq(ht);
hvl_mixchunk(ht, samples, buf1, buf2, bufmod);
buf1 += samples * bufmod;
buf2 += samples * bufmod;
loops--;
} while (loops);
2023-01-01 17:26:02 +00:00
}